fiken

Fiken Accounting Skill

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "fiken" with this command: npx skills add simonstrumse/vibelabs-skills/simonstrumse-vibelabs-skills-fiken

Fiken Accounting Skill

Norwegian accounting API integration for Fiken (fiken.no). Covers invoices, credit notes, offers, contacts, products, payments, and account balances.

Setup

Detection Order

When this skill is activated, detect the Fiken configuration in this order:

  • Centralized — Check if ~/.config/fiken-api/credentials.json exists

  • Project-local — Check if FIKEN_API_TOKEN is set in the environment (via .env )

  • No configuration — Run the First-Time Setup guide below

Using Existing Configuration

Centralized (~/.config/fiken-api/ )

If ~/.config/fiken-api/credentials.json exists:

import { getFikenClient, listAccounts } from '~/.config/fiken-api/client.mjs'

// Default account const fiken = getFikenClient() const contacts = await fiken.getContacts()

// Specific account const other = getFikenClient('account-name')

// List available accounts const accounts = listAccounts()

The centralized client handles multi-account switching, auto-includes bank account numbers, and provides account-specific config (vatType, incomeAccount, etc.) via fiken.config .

Project-local (.env )

If FIKEN_API_TOKEN is in the environment:

require('dotenv').config(); const fiken = require('./skills/fiken/scripts/client');

The project-local client reads from environment variables:

FIKEN_API_TOKEN=... # From Fiken > Innstillinger > API FIKEN_COMPANY_SLUG=... # Your company slug (from dashboard URL)

First-Time Setup

If no configuration is detected, guide the user through setup:

Step 1: Get an API token

  • Log into fiken.no

  • Go to Innstillinger > API

  • Create a new API token (costs 99 NOK/month per company)

  • Token format: {numeric-id}.{alphanumeric-key}

Step 2: Find your company slug

Step 3: Choose setup method

Use AskUserQuestion to ask the user:

  • Question: "How do you want to configure Fiken access?"

  • Option 1: Project-local (.env) — "Single project, single company. Simplest setup. Credentials in your project's .env file."

  • Option 2: Centralized (~/.config/fiken-api/) — "Multiple projects or companies. Credentials stored once in ~/.config/, used everywhere."

If project-local (.env):

  • Create .env with FIKEN_API_TOKEN and FIKEN_COMPANY_SLUG

  • Import from scripts/client.js (bundled with this skill)

  • Verify with fiken.getContacts() or fiken.isFikenConfigured()

If centralized (~/.config/fiken-api/):

  • Create ~/.config/fiken-api/credentials.json with account(s)

  • Copy the centralized client to ~/.config/fiken-api/client.mjs

  • Import via getFikenClient(accountName)

  • Verify with fiken.getContacts()

Step 4: Initialize number series

Step 5: Verify

  • Call getContacts() to confirm the token works

  • If 403 Forbidden : API module not activated (Fiken > Foretak > Tilleggstjenester)

  • If 401 Unauthorized : Bad token (regenerate at Fiken > Innstillinger > API)

Available Functions

Function Description

isFikenConfigured()

Check if env vars are set

getContacts()

List all contacts

findContactByOrgNumber(orgNr)

Find contact by Norwegian org number

createContact(input)

Create customer/supplier — returns contactId

getInvoices(page, pageSize)

List invoices (paginated)

getInvoice(id)

Get single invoice

createInvoiceDraft(input)

Create draft — returns { draftId, uuid, draftUrl }

sendInvoiceDraft(draftId, sendInput)

Two-step: finalize + dispatch — returns { invoiceId, saleId, invoiceUrl }

isInvoiceSettled(id)

Check if invoice is paid

getProducts()

List products/services

getAccounts()

Chart of accounts (~475 accounts)

getBankAccounts()

Bank accounts

nokToOre(nok) / oreToNok(ore)

Currency conversion (1 NOK = 100 ore)

delay(ms)

Rate limit helper (default 250ms)

Complete Invoice Example

require('dotenv').config(); const fiken = require('./skills/fiken/scripts/client');

// 1. Find or create contact let contact = await fiken.findContactByOrgNumber('999888777'); if (!contact) { const contactId = await fiken.createContact({ name: 'Example Company AS', email: 'post@example.no', organizationNumber: '999888777', address: { streetAddress: 'Storgata 1', city: 'Oslo', postCode: '0001', country: 'NO' }, customer: true, supplier: false }); contact = { contactId }; }

// 2. Create invoice draft const draft = await fiken.createInvoiceDraft({ type: 'invoice', // REQUIRED — see gotcha #5 issueDate: '2026-03-01', daysUntilDueDate: 14, customerId: contact.contactId, ourReference: 'Your Name', yourReference: 'Their Contact', orderReference: 'PO-2026-001', // Optional — for EHF routing invoiceText: 'Consulting services — March 2026', currency: 'NOK', // bankAccountNumber is auto-included by the client lines: [{ description: 'Consulting — project planning, 8 hours', quantity: 8, unitPrice: fiken.nokToOre(1500), // NOK 1,500/hr = 150000 ore vatType: 'HIGH', // 25% MVA (use 'OUTSIDE' if not MVA-registered) incomeAccount: '3020' // Services, high MVA (use '3220' if not MVA-registered) }] }); console.log('Draft URL:', draft.draftUrl);

// 3. Finalize + send as EHF (two-step process handled by the client) await fiken.delay(500); const invoice = await fiken.sendInvoiceDraft(draft.draftId, { method: ['ehf'], includeDocumentAttachments: true }); console.log('Invoice ID:', invoice.invoiceId); // Same as draftId console.log('Sale ID:', invoice.saleId); // Different — used for dashboard URLs console.log('Invoice URL:', invoice.invoiceUrl); // Uses saleId

VAT Quick Reference

Are you MVA-registered? This determines which vatType and income account to use.

Scenario vatType Income Account Account Name

MVA-registered, standard rate HIGH

3020

Tjenester, hoy mva-sats (25%)

MVA-registered, food/drink MEDIUM

Account varies 15% rate

MVA-registered, zero-rated EXEMPT

3120

Tjenester, fritatt for mva

Not MVA-registered OUTSIDE

3220

Tjenester, unntatt for mva

Critical distinction:

  • "Fritatt" (3120 , EXEMPT ) = zero-rated, company IS in the MVA system but rate is 0%

  • "Unntatt" (3220 , OUTSIDE ) = exempt, company is NOT in the MVA system at all

Full VAT reference with all types and account compatibility: see references/vat-guide.md

Top 7 Critical Gotchas

  1. VAT + Account must match

vatType and incomeAccount are linked. Using incompatible pairs returns 400. See the table above for correct combinations.

  1. bankAccountNumber is REQUIRED on drafts

This is YOUR bank account number (the invoice sender's) — the "pay to" account printed on the invoice. It is NOT the customer's account.

The client auto-fetches it from GET /companies/{slug}/bankAccounts (picks first active account, caches in memory). If calling the API directly, you must include bankAccountNumber (Norwegian format, not IBAN). Without it: 500: "Kontonummer mangler på faktura" . The API returns [{ bankAccountNumber, inactive }] — pick first where inactive === false .

  1. Number series — first of each type must be manual

The very first invoice/credit note/offer MUST be created in the Fiken web UI to establish the number series. After that, the API works. Error: 400: "Missing number series for drafts of type: invoice." .

  1. Money is in ore (1 NOK = 100 ore)

All monetary amounts use ore. NOK 4,998 → 499800 . Use fiken.nokToOre() and fiken.oreToNok() .

  1. type field is REQUIRED on drafts

Values: invoice , cash_invoice , offer , order_confirmation , credit_note . Omitting returns: 400: "'type' er påkrevd" .

  1. PUT replaces all fields (not a PATCH)

Contact and product updates via PUT replace the entire resource. Always include all fields. The only PATCH is on invoices: newDueDate and sentManually only.

  1. Invoice lines are immutable after finalization

Once sent, you can only change newDueDate and sentManually . To fix line items, issue a credit note and create a new invoice.

  1. Invoice sending is a TWO-STEP process (CRITICAL)

Step 1: POST /invoices/drafts/{draftId}/createInvoice — NO body, just finalizes. Step 2: POST /invoices/send — dispatches (EHF/email/auto). Without Step 2, the invoice exists but was never sent. If you pass send params to Step 1, they are silently ignored.

  1. Three different IDs per invoice

invoiceId (API operations) vs sale.saleId (dashboard URLs) vs sale.transactionId (accounting). draftId = invoiceId after finalization. The Location header from createInvoice returns saleId , NOT invoiceId .

  1. Dashboard URLs use saleId, not invoiceId

Finalized invoice URL: /foretak/{slug}/handel/salg/{saleId} — NOT /faktura/{invoiceId} .

  1. Use recipientEmail, not emailAddress

The send request field is recipientEmail . Using emailAddress is silently ignored.

All gotchas with error messages and solutions: see references/gotchas.md

Sending Methods

Method When to use

['ehf']

B2B to Norwegian companies (preferred)

['email']

Direct email delivery (requires recipientEmail )

['efaktura']

Norwegian consumer e-invoice via bank

['sms']

SMS delivery (requires mobileNumber )

['letter']

Physical letter (extra charge, Norway only)

['auto']

Fiken picks: EHF > eFaktura > SMS > Email

Multiple methods = priority list: ['sms', 'email'] tries SMS first, falls back to email.

Full sendInvoiceRequest:

{ invoiceId: 12345, // Required method: ['ehf'], // Required — array of strings includeDocumentAttachments: true, // Required recipientName: 'Name', // Optional recipientEmail: 'email@x.no', // Optional (for email method) emailSendOption: 'auto', // 'document_link' | 'attachment' | 'auto' mergeInvoiceAndAttachments: false, // Merge into single PDF organizationNumber: '999888777', // Optional (for EHF) mobileNumber: '12345678', // Optional (for SMS) message: 'Additional message text' // Optional — included in email/notification }

Dashboard URL Formats

Draft: https://fiken.no/foretak/{slug}/fakturautkast/{uuid} Invoice: https://fiken.no/foretak/{slug}/handel/salg/{saleId} Contact: https://fiken.no/foretak/{slug}/kontakt/{contactId} Credit Note: https://fiken.no/foretak/{slug}/kreditnota/{creditNoteId}

  • Domain: fiken.no (NOT app.fiken.no )

  • Draft path: fakturautkast (ONE word)

  • Drafts use UUID, not numeric ID (must GET the draft after creation)

  • Finalized invoices use saleId (from invoice.sale.saleId ), NOT invoiceId

  • The API does NOT return dashboard links — construct them from sale.saleId

Pagination

Header Description

Fiken-Api-Page

Current page (0-indexed)

Fiken-Api-Page-Size

Items per page

Fiken-Api-Page-Count

Total pages

Fiken-Api-Result-Count

Total results

Default: page=0 , pageSize=25 . Maximum: pageSize=100 .

API Limitations

  • No webhooks — must poll for changes

  • No recurring invoices — implement scheduling yourself

  • No invoice reminders/dunning via API

  • No Vipps payment integration via API

  • Cannot modify invoice lines after finalization

  • No direct payment on invoices — payments go through the underlying Sale object

Safe Testing Pattern

Drafts and products can be safely created and immediately deleted (204). This enables testing without permanent changes:

const draft = await fiken.createInvoiceDraft({ /* ... */ }); // inspect draft... // DELETE /companies/{slug}/invoices/drafts/{draftId} → 204

API Reference

  • Base URL: https://api.fiken.no/api/v2

  • Auth: Authorization: Bearer {FIKEN_API_TOKEN}

  • Rate limit: 4 req/sec (250ms between calls)

  • Content type: application/json only

  • Dates: yyyy-MM-dd format

  • Attachments: .png , .jpeg , .jpg , .gif , .pdf only

  • Total endpoints: ~110

  • API cost: 99 NOK/month per company

  • Docs: https://api.fiken.no/api/v2/docs/

Reference Files

File Contents

references/api-endpoints.md

All ~110 endpoints organized by category

references/workflows.md

Multi-line invoices, credit notes, offers, products, contacts, payments

references/gotchas.md

All 14 gotchas with error messages and solutions

references/vat-guide.md

Complete VAT types, account matrix, MVA explanation

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

bunny-net

No summary provided by upstream source.

Repository SourceNeeds Review
General

klarsprak

No summary provided by upstream source.

Repository SourceNeeds Review
General

transcribe

No summary provided by upstream source.

Repository SourceNeeds Review
General

changelog

No summary provided by upstream source.

Repository SourceNeeds Review