Create and send quotes

Learn how to create a quote and send a formal offer.

Overview

A quote is a formal offer that details how much a product or service will cost. Quotes are similar to invoices. The difference is that a quote is given to a potential customer before they decide to purchase, whereas an invoice is issued after the delivery of products or services to the customer.

Quotes have an expiry date by which the quote should be accepted or declined.

Prerequisites

To be able to create quotes, an entity must have either the tax_id or vat_id or both specified. If an entity was initially created without tax_id and vat_id, update the entity and add these values.

Create a quote

A typical quote workflow includes the following steps:

  1. Create a counterpart that represents the customer.
  2. Create one or more products to be listed in the quote.
  3. Get the applicable VAT rates based on the entity’s country and the counterpart’s country, and choose the rates for the products that will be listed in the quote.
  4. Create the quote.

1. Create a counterpart

The counterpart represents the customer (organization or individual) purchasing your products or services.

Learn how to create counterparts.

2. Create a product

The products to be listed in the quote must also be created in advance. Depending on the entity’s country, measure units may be required for products before the quote can be issued.

Learn how to create products and measure units.

As an example, here is a representation of a sample t-shirt product:

1{
2 "id": "8755c86a-d630-4920-b6fd-fd2917d87dfb",
3 "name": "T-shirt",
4 "type": "product",
5 "description": "A black cotton t-shirt, medium size",
6 "price": {
7 "currency": "EUR",
8 "value": 1000 // 10 Euro
9 },
10 "measure_unit_id": "256560f1-8d09-4d21-aadf-45b33e4b0660",
11 "smallest_amount": 1
12}

The associated measure unit is “pieces” (items):

1{
2 "id": "256560f1-8d09-4d21-aadf-45b33e4b0660",
3 "name": "pcs",
4 "description": "pieces",
5 ...
6}

3. Get VAT rates

Exception for non-supported countries, US, and Pakistan

Quote issuers in certain non-supported countries, US, and Pakistan, do not need to retrieve applicable VAT rates vis-à-vis their counterparties when creating a quote. However, Monite enables such quote issuers to provide the applicable VAT rates directly to the recipient by using the tax_rate_value field in the quote line item object. For a full list of supported countries, see Supported countries.

A quote must specify VAT rates for its line items, even if the VAT is 0%. You can call GET /vat_rates?counterpart_id={id} to get the possible VAT rates for sales to the given counterpart:

1curl -X GET 'https://api.sandbox.monite.com/v1/vat_rates?counterpart_id=0414f...5435' \
2 -H 'X-Monite-Version: 2024-05-25' \
3 -H 'X-Monite-Entity-Id: ENTITY_ID' \
4 -H 'Authorization: Bearer ACCESS_TOKEN'

The response includes the possible VAT rates (as a percentage multiplied by 100) and their IDs. Store the IDs somewhere as you will need them later.

1{
2 "data": [
3 {
4 "id": "a39a2ec3-765d-4f75-9a17-585a5d22509d",
5 ...
6 "value": 0,
7 "country": "DE"
8 },
9 {
10 "id": "51cefc6c-9f82-484e-ae6a-a3a89b507bea",
11 ...
12 "value": 700,
13 "country": "DE"
14 },
15 ...
16 ]
17}

Learn more about VAT rates.

The GET /vat_rates endpoint returns all VAT rates applicable to the entity’s VAT IDs. If the entity VAT ID is not present, the request will return only a 0% VAT rate for the entity’s country in the response.

If a counterpart is exempt from paying VAT, you can provide the reason for exemption via the optional vat_exemption_rationale field on the quote object. For example, depending on the location, counterparts in healthcare and education services may be exempt from paying VATs. You can assign a 0% VAT rate on the quote line item and provide the reason for exemption via the vat_exemption_rationale field. The reason provided will also be displayed on the quote’s PDF.

4. Create the quote

Once the counterpart and products have been created, you can create the quote for a given counterpart by calling POST /receivables with the type field in the request body set to quote.

The request body must contain the counterpart ID, line items (products, their quantities, and VAT rates), and quote expiration date, among other information. The subtotal, total, and VAT amount are not specified in the request - they will be calculated automatically based on the product price, quantity, and VAT rates.

The example below creates a quote for five items of a product:

1curl -X POST 'https://api.sandbox.monite.com/v1/receivables' \
2 -H 'X-Monite-Version: 2024-05-25' \
3 -H 'X-Monite-Entity-Id: ENTITY_ID' \
4 -H 'Authorization: Bearer ACCESS_TOKEN' \
5 -H 'Content-Type: application/json' \
6 -d '{
7 "type": "quote",
8 "counterpart_id": "0414f84c-b039-4203-b09b-e42b49245435",
9 "expiry_date": "2022-06-01",
10 "currency": "EUR",
11 "line_items": [
12 {
13 "product_id": "8755c86a-d630-4920-b6fd-fd2917d87dfb",
14 "quantity": 5,
15 "vat_rate_id": "479155c3-0995-4689-a3cf-7482ea5132a9"
16 }
17 ],
18 "vat_exempt": false,
19 "counterpart_billing_address_id": "7482ea513-4689-d630-782b6f0fs3",
20 "entity_bank_account_id": "4c6d245f-0f57-4db1-8e94-eaa6311a8e7a",
21 "counterpart_vat_id_id": "782b6f0f-0177-475c-b1e4-98dc160a656f",
22 "entity_vat_id_id": "cb6c1c38-fdae-48f8-8e51-2d50d116b882"
23 }'

Notes

  • US entities cannot use the vat_rate_id on the line_items object when creating quotes. US entities must provide custom VAT rate values using the tax_rate_value field on the line_item object.
  • When UK entities create invoices that fall under the Northern Ireland Protocol, the entity_vat_id_id field must point to a VAT ID object with type="eu_vat". Learn how to configure VAT IDs for UK entities that operate under the NI Protocol.

The successful 201 Created response contains the id assigned to the created quote, along with the calculated total (total_amount), subtotal (subtotal), VAT (total_vat_amount), and other information.

1{
2 "type": "quote",
3 "expiry_date": "2022-06-01",
4 "id": "e9a25ceb-003d-4c82-9e90-5d568ccd8c47",
5 "created_at": "2022-05-04T09:12:53.875094+00:00",
6 "updated_at": "2022-05-04T09:12:53.875103+00:00",
7 "currency": "EUR",
8 "subtotal": 5000,
9 "line_items": [
10 {
11 "quantity": 5,
12 "product": {
13 ...
14 }
15 }
16 ],
17 "total_amount": 5900,
18 "total_vat_amount": 950,
19 ...
20}

For the complete list of quote data fields and their descriptions, refer to the /receivables endpoint.

Auto-calculated amounts

Monite automatically calculates the subtotal, total, and VAT amounts for the entire quote and individual line items. These amounts are included in responses from the /receivables* endpoints. You can display these amounts in your application and use them in your business logic.

  • line_items[i].total_before_vat - line item amount including the discount but excluding VAT.
  • subtotal - quote subtotal as a sum of line item subtotals. VAT and quote-level discounts are not included, but line-item discounts are included.
  • discounted_subtotal - quote subtotal after the quote-level discount. VAT is not included.
  • total_vat_amount - total VAT amount for the entire quote.
  • total_vat_amounts - a list of VAT rates used in the document, along with the calculated total VAT amounts for each rate.
  • total_withholding_tax - if withholding_tax_rate is provided, the total withholding tax is calculated as the specified percentage of the discounted_subtotal.
  • total_amount - total quote amount after all discounts, VAT, and other taxes. It’s the sum of discounted_subtotal and total_vat_amount, minus total_withholding_tax (if any).

All monetary amounts are specified in minor currency units, such as cents or pence.

After creating invoices, you can download the invoice as a PDF or send it to the customer via email.

Download the quote as PDF

Monite automatically generates the PDF version of quotes in the counterpart language and the entity language, and returns links to these files in the file_url and original_file_url fields of quote responses:

1{
2 "type": "invoice",
3 ...
4 // Counterpart's version of a PDF quote
5 "file_language": "nl",
6 "file_url": "https://bucketname.s3.amazonaws.com/<random_UUID_1>/<random_UUID_2>.pdf",
7 ...
8 // Entity's version of a PDF quote
9 "original_file_language": "en",
10 "original_file_url": "https://bucketname.s3.amazonaws.com/<random_UUID_3>/<random_UUID_4>.pdf",
11 ...
12}

If file_url or original_file_url is null, repeat the request to GET /receivables/{quote_id} after some time.

If you need just the PDF file link without the full quote details, call GET /receivables/{quote_id}/pdf_link:

1curl -X GET 'https://api.sandbox.monite.com/v1/receivables/411dc6eb...6289b3/pdf_link' \
2 -H 'X-Monite-Version: 2024-05-25' \
3 -H 'X-Monite-Entity-Id: ENTITY_ID' \
4 -H 'Authorization: Bearer ACCESS_TOKEN'

It returns the following response:

1{
2 "file_url": "https://bucketname.s3.amazonaws.com/<random_UUID_1>/<random_UUID_2>.pdf",
3 "original_file_url": "https://bucketname.s3.amazonaws.com/<random_UUID_1>/<random_UUID_2>.pdf"
4}

Here’s how the PDF looks like:

A PDF quote
A PDF quote

The PDF file is updated automatically if the quote is changed (for example, if new line items are added to a draft quote, or if the counterpart address is changed).

PDF localization

Monite provides several built-in PDF templates for quotes. The language on the quote’s PDF depends on the language of the quote’s counterpart. If the counterpart language is absent, the quote’s PDF will be created in the entity’s language. Updating the quote’s counterpart’s language or changing the quote’s counterpart to a counterpart with a different language will also update the quote’s PDF translation. For more information, see PDF localization.

Monite formats delimiters and decimal separators on amounts displayed on the PDFs based on the entity’s country. Entities can change their default template or customize their chosen templates anytime. For more information, see PDF templates.

Verify the quote

Before a newly created quote can be issued to a counterpart, certain required information must be present on the quote. This includes information about the entity, counterpart, line items, and VAT rates on the quote. To trigger regulatory checks for a quote and check that a quote contains all required information, send a POST request to the receivables/{receivable_id}/verify endpoint as shown:

1curl -X POST 'https://api.sandbox.monite.com/v1/receivables/411dcb...289b3/verify' \
2 -H 'X-Monite-Version: 2024-05-25' \
3 -H 'X-Monite-Entity-Id: ENTITY_ID' \
4 -H 'Authorization: Bearer ACCESS_TOKEN' \

A successful response returns an object containing the missing information required on the quote. The errors object in the response shows all the missing information required on the quote. If the quote has all required information filled in, all response fields have the null value. For example, the following example shows the response for a quote without entity tax ID and counterpart tax ID information.

1{
2 "errors": {
3 "counterpart": ["tax_id"],
4 "entity": ["tax_id"],
5 "products": null,
6 "receivable": null,
7 "vat_rates": null
8 },
9 "warnings": {
10 "payment_reminders": null
11 }
12}

Send the quote via email

Once a draft quote has been finalized, you can send it to the counterpart via email. The quote will be attached as a PDF file to the email. You can also preview the quote’s email before sending it; for more information, see Preview a quote’s email.

To send a quote, call POST /receivables/{receivable_id}/send and provide the subject_text and body_text. The language of the email template used is determined by the counterpart’s language.

Aside from the counterpart’s default email address, Monite also allows you to provide a list of email addresses to which you can send the invoice using the recipients field. If the recipients field is not provided, the quote email is sent only to the counterpart’s email address.

1curl -X POST 'https://api.sandbox.monite.com/v1/receivables/e9a25c...8c47/send' \
2 -H 'X-Monite-Version: 2024-05-25' \
3 -H 'X-Monite-Entity-Id: ENTITY_ID' \
4 -H 'Authorization: Bearer ACCESS_TOKEN' \
5 -H 'Content-Type: application/json' \
6 -d '{
7 "subject_text": "New quote #{quote_number}",
8 "body_text": "Dear {contact_name},\n\nThank you for requesting a quote for your project. We are happy to provide you with our quote #{quote_number}!\n\nIf you have any questions, please don\'t hesitate to contact us at {entity_email}.",
9 "recipients": {
10 "to": ["dana@example.com"],
11 "cc": ["support@example.com", "sales@example.com"],
12 "bcc": ["exec@example.com"]
13 }
14 }'

Notes:

  • The values of the subject_text and body_text fields will replace the subject_text and body_template variables if they were used when creating your custom templates. For more information, see Create and manage email templates and Variables list. Both the subject and body texts can include variables as placeholders for quote-specific and counterpart-specific data.
  • You can also opt to attach the quote’s PDF to the email sent to the counterpart. To do this, you must update the value of the mail.attach_documents_as_pdf field to true in your partner settings. For more information, see Update partner settings.

The To email address is taken from the Counterpart object associated with the quote. This address is also returned in the counterpart_contact.email field of the Receivable object that represents the quote.

All quote emails are sent from the noreply@monite.com email address by default. You can customize the email domain name—@exampleCompany.com— by configuring a mailbox for the entity. You can also customize the email sender name and username by updating your Monite partner settings. For more information, see Update partner settings.

A 200 OK response from POST /receivables/{quote_id}/send means the email was successfully sent from the Monite email server.

Notes

  • Sending a draft quote changes its status to issued. Issued quotes can no longer be edited.
  • Only quotes in the draft, issued, and accepted statuses can be sent via email. Attempting to send a quote in the deleted, expired, or declined status will return an error.

Resend the quote

To resend an already issued quote, you can call POST /receivables/{quote_id}/send again, optionally with different subject_text and body_text templates.

Resending a quote does not change its status.