Skip to main content

A API Client for communicating with the Bunni accounting software

Project description

pyBunniApi - a Bunni Python Api Client.

Requirements

  • You need a Bunni Account.
  • You need to generate an API key for your Bunni Account.
  • Python 3.10

Installation


$ pip install pyBunniApi

Getting started with pyBunniApi


Let's start with importing the API Client

from pyBunniApi.client import Client

Once this is done we have to initialize it.

py_bunni_api = Client()
py_bunni_api.set_api_key('YOUR API KEY HERE')
py_bunni_api.set_business_id('YOUR BUSINESS ID HERE')

Optionally you can select if you want to receive all responses in a typed, or a flat dict. You can set this parameter with the following code:

py_bunni_api.use_typing(True)

The default value of this parameter is True

Receiving the contacts list


If your API key has access to 'READ' on the specific parts of contacts, we can use contacts.list to view all contacts.

contacts = py_bunni_api.contacts.list()

This will return a list of contacts. The response looks like this:

[
  {
    'id': 'co_XXXXXX',
    'companyName': 'CompanyName',
    'toTheAttentionOf': 'Berry the Bunny',
    'street': 'Carrotstreet',
    'streetNumber': '9',
    'postalCode': '1234AB',
    'city': '',
    'phoneNumber': '123456789',
    'vatIdentificationNumber': None,
    'chamberOfCommerceNumber': None,
    'color': '#112233',
    'fields': [],
    'emailAddresses': [
      'berry_the_bunny@bunni.nl'
    ]
  }
]

Receiving a list of invoices


If your API key has access to 'READ' on the invoices section, we can use invoices.list to gather a list of all invoices.

invoice_list = py_bunni_api.invoices.list()

This will return a list with all invoices, the response looks like this:

[
  {
    'id': 'in_XXXXXX',
    'invoiceDate': '2023-08-09',
    'invoiceNumber': '2023005',
    'isFinalized': True,
    'duePeriodDays': 30,
    'pdfUrl': 'https://superlongpdfurl.pdf',
    'rows': [
      {
        'description': 'This is the description of your row.',
        'quantity': 1.0,
        'unitPrice': 100
      }
    ]
  }
]

Creating an invoice PDF


This feature only generates a PDF. Said invoice will not be placed in your bookkeeping software as of now. You can however write your own piece of code that stores this pdf somewhere on your webserver, and sends it to YOUR_BUSINESS_ID@postbode.bunni.nl in order to get it automatically placed in your bookkeeping.

Anyways, this part is a little bit more spicy and requires a few more steps. Again, this only works if your API key has access to the WRITE permissions of Invoice.

First, let's start by defining our rows. A row requires four parameters. One invoice can contain varying rows. We append these bu putting rows in a list.

To create row we can initialize a Row(). The complete syntax would look like this:

row = PyBunniApi.Row(
    unit_price=12.5,  # This should be a float.
    description="This is a test description",
    quantity=5,
    tax="NL_High_21",  # This should be a string.
)

For explaining how this works, one row will be enough. The next step is to create a Contact() This can be done like this:

contact = PyBunniApi.Contact(
    company_name="The Carrot Company",
    attn='Jim Carrot',
    street='Carrot Street',
    street_number=20,
    postal_code='1122AB',
    city='Bunny Town',
    phone_number='123456789',
)

Now we can build a complete invoice using InvoicePDF() by the following manner:

invoicePdf = PyBunniApi.InvoicePDF(
    invoice_date='YYYY-MM-DD',
    invoice_number='12345.67',
    tax_mode='excl',  # This can be either `incl` or `excl`,
    design='INVOICE_DESIGN_ID',  # A little down here I'll explain how you can fetch this ID.
    contact=contact,  # We made a contact above here.
    rows=[row]
)

We now have a initialized InvoicePdf object which we can use to create a invoice (pdf) in Bunni. We can do this by using py_bunni_api.invoices.create_pdf

A complete snippet of this code would look like this:

invoice_pdf = py_bunni_api.invoices.create_pdf(invoicePdf)

This will return a single pdf url, so the expected response should look like this:

https://restpack.io/cache/pdf/069aba16b0ced81a42ecba6d7fd841885f53dd9bcac71cbbcb08756bad73e1ac

Creating an invoice###


This feature creates an invoice which is placed in your bookkeeping. It also allows you to fetch the invoice PDF.

First, let's start by defining our rows. A row requires four parameters. One invoice can contain varying rows. We append these bu putting rows in a list.

To create row we can initialize a Row(). The complete syntax would look like this:

row = PyBunniApi.Row(
    unit_price=12.5,  # This should be a float.
    description="This is a test description",
    quantity=5,
    tax="NL_High_21",  # This should be a string.
)

For explaining how this works, one row will be enough. The next step is to create a Contact() This can be done like this:

contact = PyBunniApi.Contact(
    company_name="The Carrot Company",
    attn='Jim Carrot',
    street='Carrot Street',
    street_number=20,
    postal_code='1122AB',
    city='Bunny Town',
    phone_number='123456789',
)

Now we can build a complete invoice using InvoicePDF() by the following manner:

invoicePdf = PyBunniApi.Invoice(
    external_id='Your own ID',
    invoice_date='YYYY-MM-DD',
    invoice_number='12345.67',
    tax_mode='excl',  # This can be either `incl` or `excl`,
    design='INVOICE_DESIGN_ID',  # A little down here I'll explain how you can fetch this ID.
    contact=contact,  # We made a contact above here.
    rows=[row]
)

We now have a initialized Invoice object which we can use to create a invoice in Bunni. We can do this by using py_bunni_api.invoices.create_or_update

A complete snippet of this code would look like this:

invoice_pdf = py_bunni_api.invoices.create_or_update(invoice)

This function will not return anything if your invoice object is all good. Otherwise it returns the error received from bunni.

Retreiving the list of invoice designs


For retrieving a list of invoice designs you can use invoice_designs.list A complete snippet of this code would look like this:

invoice_designs = py_bunni_api.invoice_designs.list()

The variable invoice_designs now looks like this:

[
  {
    id: "de_10XXX",
    name: "invoice THE CARROT COMPANY",
    createdOn: "2023-08-09T18:22:15.21Z"
  },
  {
    id: "de_10XXX",
    name: "New Design",
    createdOn: "2023-08-09T16:45:21.32Z"
  }
]

Retrieving a list of projects


For retrieving a list of projects we can use projects.list. A complete snippet looks like this:

projects = py_bunni_api.projects.list()

The variable projects should now contain a JSON structure like this:

[
  {
    id: "pr_17413",
    color: "#eeeeee",
    name: "Project auto voor Danny",
    externalId: "1100"
  }
]

Retieving a list of time


For retrieving a list of time objects we can use time.list. A complete snippet would look like this:

time_list = py_bunni_api.time.list()

As a result of this piece of code time_list should contain an object which looks alot like this:

[
  {
    id: "ti_29XXXX",
    date: "2023-08-10",
    duration: {
      m: 3,
      h: 5
    },
    project: {
      id: "pr_17XXX",
      color: "#123456",
      name: "Project name",
      externalId: "XXX"
    },
    description: "Time description"
  }
]

Creating or updating a time


We can create or update a time with the use of time.create_or_update. For creating or updating a time we first need to build a time object.

A time object requires two items called Duration and Project. So let's create those two first.

A duration object can be initialized like this:

duration = PyBunniApi.Duration(
    h=10, # This is a integer which stands for whole hours.
    m=30 # This is a integer which stands for whole minutes.
)

The next thing we need to setup is a Project. We can initialize one like this:

project = PyBunniApi.Project(
    id="pr_XXXXX",
    external_id="YOUR EXTERNAL ID", # Bunni documentation shows this as optional. In my experience it seems mandatory.
    color="#123456",
    name="YOUR PROJECT NAME",
)

With those two objects initialized we can create a time object. You can do that the following way:

time = PyBunniApi.TimeObject(
    date="2023-08-10",
    duration=duration,
    description="YOUR TIME DESCRIPTION",
    external_id="YOUR EXTERNAL ID",
    project=project,
)

Now that we have created all key elements we can submit it to Bunni in the following manner:

py_bunni_api.time.create_or_update(time)

A little footnote


You have made it to the end of the documentation! Well done. Please note that this project is in early development. There might be some bugs here and there. But please let me know when you find one!

Do you want to thank me, because this project helped with a puzzle you needed to solve? You can do that by buying me a coffee ;)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyBunniApi-1.2.2.tar.gz (14.6 kB view details)

Uploaded Source

Built Distribution

pyBunniApi-1.2.2-py3-none-any.whl (18.3 kB view details)

Uploaded Python 3

File details

Details for the file pyBunniApi-1.2.2.tar.gz.

File metadata

  • Download URL: pyBunniApi-1.2.2.tar.gz
  • Upload date:
  • Size: 14.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/4.0.2 CPython/3.11.8

File hashes

Hashes for pyBunniApi-1.2.2.tar.gz
Algorithm Hash digest
SHA256 f54aebfd48970bb47cc3866770690af2baa524084afca5a91fb19bc6e09b6ea0
MD5 a4b7e8caadc857ecf397a2385ca608be
BLAKE2b-256 acff9476b8f0675f4ebffca3b6f37bc166d3e3e3537e599076d29256d11ad107

See more details on using hashes here.

File details

Details for the file pyBunniApi-1.2.2-py3-none-any.whl.

File metadata

  • Download URL: pyBunniApi-1.2.2-py3-none-any.whl
  • Upload date:
  • Size: 18.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/4.0.2 CPython/3.11.8

File hashes

Hashes for pyBunniApi-1.2.2-py3-none-any.whl
Algorithm Hash digest
SHA256 61801d5af76d43a799a599f8b308c19fde81675529d558781a0291351cb81b78
MD5 6c11486ed783d60b251d9d6fe3d7f382
BLAKE2b-256 bdedd0e391800c9c22eb0dd1bb104055dbf6cc6e1761041f83a9711f28c200a8

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page