Status: Public API surface. Source maintained in
a0b1c0/gPdfunderdoc/contracts/api/; website and docs-site copies are synchronized publication outputs. Last updated: 2026-05-18This document defines what callers of the gPdf HTTP API can rely on. It does not describe internal control-plane behaviour, infrastructure, or any field not listed here. Anything not documented here is not part of the public contract and may change without notice.
gPdf turns a JSON request into a PDF document. There are three public APIs, each tuned for a different integration shape:
| API | Endpoint | When to use |
|---|---|---|
| JSON Render | POST /api/v1/pdf/render |
You describe pages, elements, coordinates, tables, and pagination yourself. Best for designers, custom reports, and one-off layouts. |
| Template Render | POST /api/v1/template-render |
You only send template_id + business data. Best for ERP, OMS, WMS, and any system that wants a stable contract per document type. |
| E-Invoice Render | POST /api/v1/e-invoice/render |
You need a Factur-X / ZUGFeRD compliant PDF/A-3b with embedded CII XML. |
A simple decision tree:
- "I want pixel control over the layout." → JSON Render
- "My team agreed on a template name and a list of fields." → Template Render
- "I need an EU-mandate-compliant electronic invoice." → E-Invoice Render
JSON Render and Template Render return application/pdf on success.
E-Invoice Render returns application/pdf for delivery.mode = inline_pdf
and a JSON job descriptor for delivery.mode = object. All three return
application/json on error, share the same authentication, and share the
same error-code namespace.
Machine-readable spec: the full API contract is published as an OpenAPI 3.1 document at
/openapi.json(YAML mirror at/openapi.yaml). Use it to generate SDKs (openapi-generator), import into Postman / Insomnia, drive IDE autocomplete, or hand to an AI coding assistant — the OpenAPI is the canonical contract this document narrates.
Every request uses:
POST /api/v1/<route>
Host: api.gpdf.com
Content-Type: application/json
Authorization: Bearer <token>
X-Request-Id: <optional-client-id>Every successful response carries:
Content-Type: application/pdfContent-Disposition: inline; filename="..."(default) orattachment; filename="..."whenoutput.mode = "file"X-Request-Id: <echoed-or-generated>
Every error response carries:
Content-Type: application/jsonX-Request-Id: <echoed-or-generated>
The error body is always:
{
"error": true,
"code": "API-002",
"message": "x must be >= 0",
"req_id": "7f7d2f5a-4cb0-4c4e-b6ef-8f6d3e0b1fd8"
}| Field | Type | Notes |
|---|---|---|
error |
boolean |
Always true on the error envelope. |
code |
string |
Public error code. See §6.1. |
message |
string |
Human-readable explanation. Some auth and system errors return a redacted message. |
req_id |
string |
Mirrors X-Request-Id. Always present. Use it when filing support tickets. |
The following are intentionally not part of the public contract:
- Internal storage layout, queue topology, or any specific cloud service used to deliver responses.
- Token issuance, billing, quota provisioning, and policy management. Those happen in the gPdf Console, not over this API.
- Template authoring (how to design and publish a template). See the internal
template-authoring.en.mdif you maintain templates. - Beta endpoints not listed in this document.
gPdf runs two public environments. They are isolated: tokens, templates, and e-invoice jobs do not cross over.
| Environment | Base URL | Purpose |
|---|---|---|
| Production | https://api.gpdf.com |
Live traffic. SLAs apply. |
| Test | https://api-test.gpdf.com |
Pre-integration sandbox. Same API shape, separate tokens, separate templates, no SLA. |
Build your client so the base URL is configuration, not a constant. Most
integrations read it from an environment variable like GPDF_BASE_URL.
Every render endpoint requires a Bearer token:
Authorization: Bearer sk_live_<YOUR_API_KEY>Rules:
- The token is opaque. Do not parse it.
- Tokens are environment-scoped. A test token will be rejected by production with
API-102. - Tokens may carry policy constraints (max pages per request, allowed PDF/A profiles, allowed e-invoice standards). Constraint violations return
API-002with the offending field named inmessage. - A revoked or expired token returns
API-103with a redacted message. Treat both as "rotate or contact support".
The single endpoint that does not require authentication is
GET /api/v1/e-invoice/capabilities. See the dedicated
E-invoice API reference.
For AI coding assistants such as Cursor, Copilot, custom GPTs, and developers
who want to quickly test or debug DocumentRequest JSON payloads without
registering or managing API keys, gPdf provides a public sandbox proxy endpoint.
POST /api/playground?endpoint=pdf-render
Host: gpdf.com
Content-Type: application/json
Accept: application/pdfNote
Sandbox guidelines and policies:
- No Authorization header required: this trial sandbox automatically binds a secure developer key at the CDN edge. Do not include an
Authorizationheader. - Development and trial only: this endpoint is restricted to local debugging, layout validation, and interactive AI evaluation. It is not for production workloads.
- Automatic trial watermark: PDFs generated through this trial sandbox are stamped with a semi-transparent
gpdf.com Sandbox - Test Onlywatermark to prevent spoofing and commercial misuse. - Rate limits and payload boundaries:
- Rate limit: maximum 30 requests per minute per IP address. Exceeding this returns
429 Too Many Requests. - Max request size: request JSON is limited to 256 KB.
- Output format: the response is an
application/pdfbinary stream.
- Rate limit: maximum 30 requests per minute per IP address. Exceeding this returns
- Commercial use: for clean production traffic without watermarks, higher throughput, or advanced features, register in the gPdf Console and use a dedicated live token.
Every request gets a request ID. You may supply one via the X-Request-Id
header; if you don't, gPdf generates one. It is echoed in every response —
both success and error — and is included in error.req_id.
Recommended client behaviour:
- Generate a UUID v4 per outbound request and pass it as
X-Request-Id. - Log the request ID alongside your application's correlation ID.
- Quote it verbatim when reporting issues.
curl -X POST "https://api.gpdf.com/api/v1/pdf/render" \
-H "Authorization: Bearer $GPDF_TOKEN" \
-H "Content-Type: application/json" \
-H "X-Request-Id: $(uuidgen)" \
--data-binary @request.json \
--output out.pdfgPdf does not currently publish rate-limit headers (X-RateLimit-*,
Retry-After) or an idempotency-key contract. This is a deliberate omission
in the v1 surface; future versions may add them.
Recommended client behaviour today:
- Cap concurrency per token at a number you negotiated with your gPdf account contact. Most production tokens are sized for sustained 5-20 req/s; bursts above that should be queued client-side.
- On
5xxor network error, retry with exponential backoff (initial 500 ms, max 5 s, max 3 attempts). - On
4xx, do not retry. The request will fail the same way every time. Inspectcodeandmessageand fix the request. - For at-most-once semantics on PDF generation, deduplicate on your side before calling. The API does not detect duplicate submissions.
The fastest way to verify your token, the route, and PDF byte handling is the
five-second request below. Save it as quickstart.json:
{
"pages": [
{
"size": "label_100_150",
"elements": [
{
"type": "text",
"x": 10,
"y": 18,
"content": "Hello gPdf"
}
]
}
]
}Then call the API:
curl -X POST "https://api.gpdf.com/api/v1/pdf/render" \
-H "Authorization: Bearer $GPDF_TOKEN" \
-H "Content-Type: application/json" \
-H "X-Request-Id: quickstart-001" \
--data-binary @quickstart.json \
--output quickstart.pdfYou should get a 100 × 150 mm one-page PDF on disk. If you don't:
| Symptom | Likely cause | Fix |
|---|---|---|
401 with API-101 |
Missing or malformed Authorization header |
Confirm the header is exactly Bearer <token>. |
401 with API-102 |
Token rejected | Confirm the token belongs to the environment you are calling. |
400 with API-001 |
Body is not valid JSON | Check the file with jq . or a JSON linter. |
400 with API-002 |
Body parsed but failed validation | Read message — it names the offending field. |
200 with empty body |
Saved with --output but the response is JSON |
Drop --output, re-run; you will see a JSON error envelope. |
After the minimum request succeeds, move on to:
- §4 JSON Render API for full layout control.
- §5 E-Invoice Render API for compliance documents.
- The Template Render API (
template-api.en.md) fortemplate_id + dataintegrations.
POST /api/v1/pdf/render is the lower-level rendering endpoint. The request
body is a single DocumentRequest JSON object that describes pages, elements,
styles, and pagination behaviour explicitly. The caller has full control.
If you want a higher-level integration where you only send a template ID and
business data, use POST /api/v1/template-render (see template-api.en.md).
| Property | Value |
|---|---|
| Method | POST |
| Path | /api/v1/pdf/render |
| Auth | Required — Authorization: Bearer <token> |
Request Content-Type |
application/json |
| Success | 200, Content-Type: application/pdf |
| Error | 4xx / 5xx, Content-Type: application/json |
curl -X POST "https://api.gpdf.com/api/v1/pdf/render" \
-H "Authorization: Bearer $GPDF_TOKEN" \
-H "Content-Type: application/json" \
-H "X-Request-Id: $(uuidgen)" \
--data-binary @request.json \
--output out.pdf{
"settings": { },
"layers": { },
"header": { },
"footer": { },
"pages": [ ]
}| Field | Type | Required | Notes |
|---|---|---|---|
settings |
Settings |
No | Global defaults, metadata, output mode, PDF/A profile. See §4.14. |
layers |
Layers |
No | Background / watermark / stamp. See §4.13. |
header |
Section |
No | Global page header. See §4.12. |
footer |
Section |
No | Global page footer. See §4.12. |
pages |
Page[] |
Yes | One or more pages. |
The smallest valid request:
{
"pages": [
{
"size": "a4",
"elements": [
{ "type": "text", "x": 10, "y": 20, "content": "Hello gPdf" }
]
}
]
}When you omit settings, gPdf applies safe defaults for fonts, strokes,
fills, and output mode. You do not need to write a 200-line DocumentRequest
to get a usable PDF.
A page is described by its size and the elements it contains. Body elements
position themselves in millimetres from the page's top-left corner unless
page_margin is configured.
| Field | Type | Required | Notes |
|---|---|---|---|
size |
string |
One of size or width+height |
Named preset. Case-insensitive. |
width |
number |
One of size or width+height |
Custom page width in mm. Must satisfy 10 <= width <= 2000. |
height |
number |
One of size or width+height |
Custom page height in mm. Must satisfy 10 <= height <= 2000. |
margin |
PageMargin |
No | Per-page margin override. |
elements |
Element[] |
No | Body elements. May be empty. |
Rules:
sizeandwidth/heightare mutually exclusive on the same page. Providing both returnsAPI-002.- A page without
sizemust provide bothwidthandheight. - Custom
width/heightvalues are in millimetres and are limited to10 <= value <= 2000per side. This covers standard paper, labels, engineering drawings, and common posters while catching common unit mistakes such as inches or pixels submitted as millimetres.
| Preset | Dimensions | Typical use |
|---|---|---|
a4 |
210 × 297 mm |
Default office page in most non-US locales. |
a6 |
105 × 148 mm |
Postcards, small notices. |
letter |
215.9 × 279.4 mm |
US default. |
legal |
215.9 × 355.6 mm |
US legal documents. |
label_100_100 |
100 × 100 mm |
Square labels. |
label_100_150 |
100 × 150 mm |
Most common shipping label. |
label_4_6_in |
101.6 × 152.4 mm |
US 4×6" shipping label. |
{
"pages": [
{ "size": "a4", "elements": [] },
{ "size": "letter", "elements": [] },
{ "width": 100, "height": 150, "elements": [] }
]
}Configure margins globally (settings.page_margin) or per page
(pages[].margin):
{
"settings": {
"page_margin": { "top": 10, "right": 12, "bottom": 10, "left": 12 }
},
"pages": [
{
"size": "letter",
"margin": { "top": 8, "right": 10, "bottom": 12, "left": 10 },
"elements": []
}
]
}Behaviour:
- Without
page_margin, body elements use absolute page coordinates. - Once
page_marginis set, body elementx/ybecome relative to the content box (the area inside the margins). - Elements that overflow the content box return
API-002. There is no automatic clipping. headerandfooteralways use absolute page coordinates and ignore margins.- Auto-paginated overflow continues from the top of the next page's content box.
- All coordinates and lengths are in millimetres (mm).
- The origin is the top-left corner of the page (or the content box if
page_marginis set). - The X axis goes right; the Y axis goes down.
- See §6.5 for
rotationrules per element.
The pages[].elements array (and the body of header, footer,
layers.background, layers.stamp) contains element objects. Every element
has a type discriminator.
type |
Section | Notes |
|---|---|---|
text |
§4.6 | Plain, rich spans, or block text. |
barcode |
§4.7 | 2D matrix and 1D linear formats. |
image |
§4.8 | Asset reference or inline base64. |
line |
§4.9 | Single line segment. |
rect |
§4.9 | Rectangle, optionally rounded. |
circle |
§4.9 | Circle by centre + radius. |
ellipse |
§4.9 | Ellipse by centre + radii. |
polygon |
§4.9 | Closed polygon from a point list. |
link |
§4.9 | Standalone clickable hotspot. |
table |
§4.10 | Tabular data with headers, spans, pagination. |
stack |
§4.11 | Vertical composition of a table followed by trailing blocks. |
Common fields shared by most elements:
| Field | Type | Notes |
|---|---|---|
z_index |
number |
Stacking order. Default 0. Higher draws on top. |
comment |
string |
Free-form note. Not rendered. |
rotation |
number |
See per-element rules. Most elements support 0/90/180/270; text and image accept any integer angle. |
link |
LinkSpec |
Make the element clickable. See §4.9.6. |
Hyperlink modes:
- Attach
linkto an element (text,barcode,line,rect,circle,ellipse,polygon,image). - Use a standalone
type: "link"hotspot when you need to overlay a clickable region.
x_anchor aligns elements relative to a reference edge instead of an absolute
X. It is supported on text, barcode, rect, image, and link. It is
not supported on line, circle, ellipse, polygon, table,
stack, or block-text content.
{
"type": "text",
"x_anchor": { "reference": "content_right", "offset": 8 },
"y": 12,
"content": "$1,235.85",
"style": {
"width": 24,
"text_align": "right"
}
}| Reference | Meaning |
|---|---|
page_left |
Left page edge. |
page_right |
Right page edge. |
content_left |
Left edge of the content box. Falls back to the page edge if no margin is set. |
content_right |
Right edge of the content box. |
table_left |
Left edge of the parent table. Only valid inside stack > block. |
table_right |
Right edge of the parent table. Only valid inside stack > block. |
Resolved X:
- Left references:
resolved_x = reference + offset - Right references:
resolved_x = reference - offset - element_width
Rules:
xandx_anchorare mutually exclusive. Sending both returnsAPI-002.- When using
x_anchor, the element must have a width:text(plain orspansshorthand):style.widthtext(block):frame.widthbarcode,rect,image,link: their ownwidthfield.
The text element accepts three input forms:
| Form | Use when |
|---|---|
| Plain text shorthand | One short string in one style. |
spans rich text |
One paragraph mixing multiple inline styles. |
| Block text | Multi-paragraph layout, lists, page breaks, variables, or pagination control. |
All three pass through the same validation and rendering pipeline. Block text is the most expressive.
Required fields for any form:
y(number).content(string,{ spans }, or{ blocks }).- One of
xorx_anchor.
{
"type": "text",
"x": 18,
"y": 18,
"content": "Hello gPdf"
}With styling:
{
"type": "text",
"x": 18,
"y": 18,
"content": "Invoice #INV-2026-001",
"style": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 12,
"font_weight": "bold",
"color": "#111827",
"width": 90,
"text_align": "left",
"line_height": 1.25,
"letter_spacing": 0.2
}
}spans is a single paragraph composed of style runs. Each span inherits
the element-level style and overrides only what it sets — typical use
cases are price lines, badges, and inline footnote markers.
{
"type": "text",
"x": 18,
"y": 30,
"content": {
"spans": [
{ "text": "Total ", "style": { "color": "#6b7280" } },
{ "text": "USD ", "style": { "color": "#6b7280", "font_size": 9 } },
{ "text": "1,248.50",
"style": { "font_weight": "bold", "font_size": 14, "color": "#111827" } },
{ "text": " (incl. tax)", "style": { "color": "#6b7280" } },
{ "text": "*",
"style": { "script": "superscript", "color": "#dc2626" } }
]
},
"style": {
"font_family": "NotoSans-Regular",
"font_size": 11,
"width": 90
}
}Block text uses an explicit document tree of blocks and inlines. The
example below stitches together every common feature in one element — a
heading paragraph, a justified body paragraph mixing bold / coloured /
italic runs, an ordered list whose items each carry their own bold
emphasis, and a right-aligned footer paragraph that prints page / total_pages. The frame uses overflow: "paginate" so the article flows
across as many pages as it needs.
{
"type": "text",
"x": 18,
"y": 38,
"frame": {
"width": 174,
"overflow": "paginate"
},
"defaults": {
"run": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 10.5,
"color": "#1f2937"
},
"paragraph": {
"align": "left",
"direction": "auto",
"line_height": 1.45
}
},
"content": {
"blocks": [
{
"type": "paragraph",
"inlines": [
{
"type": "text",
"text": "Quarterly Review",
"style": { "font_size": 16, "font_weight": "bold", "color": "#111827" }
}
]
},
{
"type": "paragraph",
"style": { "align": "justify" },
"inlines": [
{ "type": "text", "text": "Revenue grew " },
{ "type": "text", "text": "23% year-over-year",
"style": { "font_weight": "bold", "color": "#0f766e" } },
{ "type": "text", "text": " on the back of three drivers — enterprise expansion, the new self-serve tier, and improved retention. The next two quarters will focus on the " },
{ "type": "text", "text": "EMEA rollout",
"style": { "font_style": "italic" } },
{ "type": "text", "text": "." }
]
},
{
"type": "list",
"list": { "kind": "ordered" },
"items": [
{ "blocks": [{ "type": "paragraph", "inlines": [
{ "type": "text", "text": "Enterprise pipeline doubled to " },
{ "type": "text", "text": "$48M qualified ARR",
"style": { "font_weight": "bold" } },
{ "type": "text", "text": "." }
] }] },
{ "blocks": [{ "type": "paragraph", "inlines": [
{ "type": "text", "text": "Self-serve activation reached " },
{ "type": "text", "text": "61%", "style": { "font_weight": "bold" } },
{ "type": "text", "text": " — three points above target." }
] }] },
{ "blocks": [{ "type": "paragraph", "inlines": [
{ "type": "text", "text": "Net retention held at " },
{ "type": "text", "text": "118%", "style": { "font_weight": "bold" } },
{ "type": "text", "text": ", a record for the company." }
] }] }
]
},
{
"type": "paragraph",
"style": { "align": "right" },
"inlines": [
{ "type": "text", "text": "Page " },
{ "type": "variable", "name": "page", "scope": "system" },
{ "type": "text", "text": " / " },
{ "type": "variable", "name": "total_pages", "scope": "system" }
]
}
]
}
}Restriction: a single
textelement cannot mix an explicitpage_breakblock withsystem.pageorsystem.total_pagesvariables. Page numbers are evaluated before pagination, so the renderer rejects the combination at validation time. Useframe.overflow = "paginate"to break across pages when content overflows (as above), or usepage_breakblocks with static text only.
Top-level fields:
| Field | Type | Required | Notes |
|---|---|---|---|
type |
"text" |
Yes | |
y |
number |
Yes | |
x or x_anchor |
One of | See §4.5.1. | |
rotation |
integer |
No | Any integer angle. |
z_index |
number |
No | |
comment |
string |
No | |
link |
LinkSpec |
No | Element-level link. Mutually exclusive with any inline link. |
style |
TextStyle |
No | For plain or spans form. |
frame |
BlockTextFrame |
No | For block form. |
defaults |
BlockTextDefaults |
No | Default run / paragraph / frame for the block tree. |
content |
string | { spans } | { blocks } |
Yes | The text content in one of the three forms. |
{
"type": "paragraph",
"style": {
"align": "justify",
"direction": "auto",
"line_height": 1.35
},
"inlines": [
{ "type": "text", "text": "Line 1: " },
{ "type": "variable", "name": "page", "scope": "system" }
]
}paragraph.style fields:
- Currently rendered (validated and applied at runtime):
align:left | center | right | justifydirection:auto | ltr | rtlline_height(number)
- Accepted but not yet wired into the runtime renderer (validator
rejects them with
paragraph layout features … not wired; the field list is reserved so future API additions can land without a contract break):space_before,space_after(mm)indent_left,indent_right,indent_first_line,hanging_indent(mm)keep_together,keep_with_next(boolean)widow_orphan_control(boolean)tabs(array)
{
"type": "list",
"list": { "kind": "bullet" },
"items": [
{
"blocks": [
{
"type": "paragraph",
"inlines": [
{ "type": "text", "text": "First item — " },
{ "type": "text", "text": "with bold emphasis",
"style": { "font_weight": "bold" } }
]
}
]
},
{
"blocks": [
{
"type": "paragraph",
"inlines": [{ "type": "text", "text": "Second item, first paragraph." }]
},
{
"type": "paragraph",
"inlines": [{ "type": "text", "text": "An item may carry multiple paragraphs." }]
}
]
},
{
"blocks": [
{
"type": "paragraph",
"inlines": [{ "type": "text", "text": "Third item." }]
}
]
}
]
}list.list accepts only kind (ordered | bullet) at runtime today.
Spacing / numbering controls (marker_gap, item_spacing, start_at,
continuation rules) are reserved in the schema but the renderer is not yet
wired for them — validator rejects them with list spacing/continuation features … not wired. Stick to kind for runnable examples until a
release note opens the rest.
list is only allowed in the full text profile (see §4.6.4). It is
rejected inside header, footer, layers, table cells, and barcode text.
{ "type": "page_break" }page_break is the only way to force a page break inside block text. The
older \f string convention is no longer supported.
Four inline node types are public:
| Type | Example |
|---|---|
text |
{ "type": "text", "text": "Hello", "style": { "font_weight": "bold" } } |
variable |
{ "type": "variable", "name": "page", "scope": "system" } |
line_break |
{ "type": "line_break" } |
tab |
{ "type": "tab" } |
variable.scope may be:
system— page numbers and totals. JSON Render only resolvespageandtotal_pages.binding— values supplied by the template-data pipeline. Validation fails in JSON Render. Use Template Render for data substitution.computed— derived values. Validation fails in JSON Render today.
run-level fields:
font_family,font_sizefont_weight:normal | medium | semibold | boldfont_style:normal | italicfont_mode:strict | prefercolor,opacity,letter_spacingscript:normal | superscript | subscriptbackground,decoration,link_style
Rules:
font_modecannot appear without a same-levelfont_family.font_mode = "auto"is not a public input. Auto mode is implicit when no font is declared anywhere in the inheritance chain.strictfailure (declared font cannot cover the text) returnsAPI-002.autoorprefertotal fallback failure returnsAPI-504.
frame fields:
width,height(mm)vertical_align:top | middle | bottomoverflow:visible | clip | ellipsis | paginateshrink_to_fit(boolean)min_font_size(number)padding,border,backgroundcolumns,column_gap
Rules:
frameis rejected inside table cells and barcode text.rotation != 0cannot combine withframe.overflow = "paginate".frame.heightcannot combine withpage_break.frame.overflow ∈ { clip, ellipsis }cannot combine withpage_break.frame.overflow = "paginate"cannot combine withshrink_to_fit = true.header/footer/ layer text cannot usepaginateor multi-column.
The exact subset of text features available depends on where the text
appears. There is no profile field in the JSON; the profile is implied by
container.
| Profile | Used in | Allowed blocks | Allowed inlines |
|---|---|---|---|
| Full | pages[].elements[] (top-level body text) |
paragraph, list, page_break |
text, variable, line_break, tab |
| Section | header, footer, layers.background, layers.watermark, layers.stamp |
paragraph |
text, variable, line_break, tab |
| Table | Inside table cells |
paragraph |
text, variable, line_break |
| Barcode | Inside barcode_text |
paragraph |
text, variable, line_break |
All three restricted profiles reject list and page_break. Table and
barcode profiles additionally reject frame, tab, and inline link. The
section profile keeps frame available but with two carve-outs:
frame.overflow = "paginate" and multi-column (frame.columns > 1) are not
allowed inside a header / footer / layer (the parent section is not itself a
paginating context).
Frame fields enter validation through the plain text.style shorthand too —
style.width / style.height / style.vertical_align / style.text_overflow
/ style.shrink_to_fit / style.min_font_size are coerced into a shadow
frame for contract checking. Setting any of these on a barcode_text.style
or table-cell text style therefore produces the same frame is not allowed
rejection as a literal frame: { ... } block.
2D / matrix code (square module grid):
{
"type": "barcode",
"x": 18,
"y": 34,
"format": "qrcode",
"content": "https://gpdf.com/docs/api-reference/#47-barcode",
"width": 28,
"height": 28,
"style": {
"color": "#111111",
"background_color": "#FFFFFF"
},
"barcode_text": {
"enabled": true,
"position": "bottom",
"offset": 1.5,
"style": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 8,
"color": "#374151",
"text_align": "center"
}
}
}1D / linear code (rectangular bars). The same shape works for any format
in the linear group below — only format and content change:
{
"type": "barcode",
"x": 18,
"y": 70,
"format": "code128",
"content": "INV-2026-001",
"width": 60,
"height": 16,
"style": {
"color": "#000000",
"background_color": "#FFFFFF"
},
"barcode_text": {
"enabled": true,
"position": "bottom",
"offset": 1.0,
"style": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 8,
"color": "#111111",
"text_align": "center"
}
}
}Required fields:
y,format,content,width,height.- One of
xorx_anchor.
Optional: style, options, barcode_text, rotation, z_index,
comment, link.
Rules:
rotationaccepts only0,90,180,270.barcode_textinherits the same rotation.formatis case-insensitive.-and_are equivalent separators.- 2D / matrix codes encode as a module matrix; 1D / linear codes encode as bars;
maxicodeuses a hexagonal grid.
Supported format values (current build):
- 2D / matrix:
qrcode(qr),microqr(micro-qr),pdf417,micropdf417,datamatrix(data-matrix),gs1datamatrix,aztec,maxicode,gs1qrcode - 1D / linear:
code128,code128a,code128b,code128c,gs1128,code39,code93,codabar,ean8,ean13,upca,upce,itf(interleaved2of5),itf14,gtin8,gtin12,gtin13,gtin14,isbn,sscc - Other:
msi,msi10,msi11,msi1010,msi1110,upus10,uspsimb,upcacomposite,upcecomposite
{
"type": "image",
"x": 4,
"y": 8,
"width": 33,
"height": 11,
"asset": "aitrack-1",
"format": "jpg"
}Required fields:
y,width,height.- One of
xorx_anchor.
Optional: rotation, z_index, comment, link.
Image source — exactly one of:
- Shorthand: top-level
asset(with optional top-levelformat). - Explicit: top-level
sourceobject.
{ "type": "image", "x": 4, "y": 8, "width": 33, "height": 11,
"source": { "kind": "asset", "key": "aitrack-1", "format": "jpg" } }{ "type": "image", "x": 4, "y": 8, "width": 33, "height": 11,
"source": { "kind": "base64", "format": "jpg",
"payload": "/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAMCAgICAgMCAgIDAwMDBAYEBAQEBAgGBgUGCQgKCgkICQkKDA8MCgsOCwkJDRENDg8QEBEQCgwSExIQEw8QEBD/2wBDAQMDAwQDBAgEBAgQCwkLEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBD/wAARCAAIAAgDAREAAhEBAxEB/8QAFAABAAAAAAAAAAAAAAAAAAAAB//EABQQAQAAAAAAAAAAAAAAAAAAAAD/xAAVAQEBAAAAAAAAAAAAAAAAAAAHCP/EABQRAQAAAAAAAAAAAAAAAAAAAAD/2gAMAwEAAhEDEQA/ACpaAPf/2Q==" } }Rules:
assetandsourceare mutually exclusive.source.kindacceptsassetandbase64.- The
payloadforbase64is the raw base64 content without adata:image/...;base64,prefix. Data URIs are rejected. - Supported formats:
jpg,jpeg,png,webp,svg. rotationaccepts any integer angle (e.g.45,-30).- See §6.2 for image and total request body size limits.
Shape elements share the same stroke, fill, link, and stacking rules. Each
may attach an optional link: LinkSpec.
{
"type": "line",
"x1": 4, "y1": 99,
"x2": 96, "y2": 99,
"stroke": {
"color": "#000000",
"width": 0.4,
"dash": { "preset": "solid" }
}
}Stroke fields fall back through settings.defaults.stroke → service defaults
when omitted (see §4.16).
{
"type": "rect",
"x_anchor": { "reference": "content_right", "offset": 6 },
"y": 20,
"width": 60,
"height": 20,
"fill": { "color": "#FFFFFF" },
"stroke": { "color": "#222222", "width": 0.6 },
"corner_radius": 2
}{
"type": "circle",
"cx": 40, "cy": 40, "r": 12,
"fill": { "color": "#E6F4FF" },
"stroke": { "color": "#2B6CB0", "width": 0.5 }
}{
"type": "ellipse",
"cx": 70, "cy": 40, "rx": 16, "ry": 10,
"rotation": 0,
"fill": { "color": "#FFF7E6" },
"stroke": { "color": "#C05621", "width": 0.5 }
}{
"type": "polygon",
"points": [
{ "x": 20, "y": 80 },
{ "x": 35, "y": 60 },
{ "x": 50, "y": 80 },
{ "x": 40, "y": 95 }
],
"fill": { "color": "#F0FFF4" },
"stroke": { "color": "#2F855A", "width": 0.5 }
}A LinkSpec is reused everywhere a hyperlink is allowed:
{
"target": { "type": "url", "url": "https://gpdf.com/docs/api-reference/#496-link-spec-and-standalone-link" },
"alt": "Open the official site",
"padding": 1.0,
"border": { "color": "#1A202C", "width": 0.3 }
}LinkTarget variants:
- URL:
{ "type": "url", "url": "https://..." } - In-document jump:
{ "type": "page", "page": 2, "x": 10, "y": 20 }
Rules:
- URL schemes allowed:
http://,https://,mailto:,tel:. Others returnAPI-002. - URL strings are trimmed before being written to the PDF annotation.
pageis 1-indexed and must not exceed the request's page count.paddingandborder.widthmust be finite and>= 0.border.colormust be a valid hex colour if provided.
Standalone link element:
{
"type": "link",
"x_anchor": { "reference": "content_left", "offset": 10 },
"y": 10,
"width": 40,
"height": 8,
"target": { "type": "url", "url": "https://gpdf.com/docs/api-reference/#496-link-spec-and-standalone-link" },
"alt": "Open website"
}Use the standalone form for clickable region overlays not bound to a specific element.
A table renders tabular data with optional grouped headers, row headers, cell merging, alternate row fills, configurable grids, and page-break behaviour.
{
"type": "table",
"x": 12,
"y": 24,
"width": 180,
"columns": [
{ "key": "sku", "header": "SKU", "width": { "mode": "fixed", "value": 30 } },
{ "key": "name", "header": "Name", "width": { "mode": "auto" } },
{ "key": "qty", "header": "Qty", "width": { "mode": "fixed", "value": 18 } },
{ "key": "amount", "header": "Amount", "width": { "mode": "fixed", "value": 30 },
"cell": { "text": { "text_align": "right" } } }
],
"rows": [
{ "sku": "A001", "name": "Widget", "qty": 2, "amount": "$120.00" },
{ "sku": "A002", "name": "Gadget", "qty": 1, "amount": "$60.00" }
],
"header": {
"show": true,
"repeat_on_page_break": true,
"cell": { "fill": { "color": "#F3F4F6" }, "text": { "font_weight": "bold" } }
},
"grid": {
"horizontal": { "color": "#D1D5DB", "width": 0.2 },
"vertical": false
}
}Top-level fields:
| Field | Type | Required | Notes |
|---|---|---|---|
x |
number |
Yes | Top-left X (mm). |
y |
number |
Yes | Top-left Y (mm). |
width |
number |
No | Total table width. Required when any column uses percent or auto. |
columns |
TableColumn[] |
Yes | Column definitions. At least 1. |
rows |
TableRow[] |
Yes | Row data. May be empty. |
cell |
TableCellStyle |
No | Default cell style for the whole table. |
header |
TableHeaderConfig |
No | Column header configuration. |
row_header |
TableZoneConfig |
No | Row-header zone configuration. |
body |
TableBodyConfig |
No | Body-zone configuration. |
grid |
TableGridConfig |
No | Grid lines. |
pagination |
TablePaginationConfig |
No | Page-break behaviour. |
z_index, comment |
No | Common element fields. |
{
"key": "amount",
"header": "Amount",
"width": { "mode": "fixed", "value": 30 },
"role": "data",
"cell": { "text": { "text_align": "right" } },
"header_cell": { "text": { "font_weight": "bold" } }
}| Field | Type | Required | Notes |
|---|---|---|---|
key |
string |
Yes | Unique within the table. |
header |
string |
No | Default empty string. |
width |
TableColumnWidth |
Yes | One of `fixed |
role |
string |
No | data (default) or row_header. |
cell |
TableCellStyle |
No | Per-column body cell style. |
header_cell |
TableCellStyle |
No | Per-column header cell style. |
TableColumnWidth:
[
{ "mode": "fixed", "value": 30 },
{ "mode": "percent", "value": 25 },
{ "mode": "auto" }
]Rules:
columns[].keymust be unique.role = "row_header"columns must be contiguous on the left. You may have multiple row-header columns.
rows is an array of objects keyed by columns[].key.
Shorthand cell (scalar value):
{ "name": "Apple", "qty": 2, "enabled": true, "note": null }Complex cell:
{
"group": {
"content": "Fruit",
"row_span": 2,
"col_span": 1,
"style": { "text": { "font_weight": "bold" } },
"link": { "target": { "type": "url", "url": "https://gpdf.com/docs/api-reference/#4102-row-and-cell" } }
}
}Complex cell fields:
| Field | Type | Notes |
|---|---|---|
content |
string | number | boolean | null | BlockTextContent |
Cell content. Scalars, or block text under the table profile (§4.6.4). |
row_span |
integer |
>= 1. Merge downward. |
col_span |
integer |
>= 1. Merge rightward. |
style |
TableCellStyle |
Per-cell override. |
link |
LinkSpec |
Cell-level link. Only on complex cells. |
Rules:
nullrenders as an empty string.booleanrenders as"true"/"false".- Spans cannot exceed the table boundary or overlap each other. Violations return
API-002. - Rows containing a key not declared in
columns[].keyreturnAPI-002(no silent ignore).
{
"padding": { "x": 1, "y": 1 },
"text": { "font_size": 9, "color": "#111111" },
"fill": { "color": "#FFFFFF" },
"content_offset_x": 1.5,
"content_offset_y": 0.5,
"borders": {
"top": false,
"right": { "color": "#111111", "width": 0.2 },
"bottom": { "color": "#111111", "width": 0.2 },
"left": false
}
}Borders may be false (suppress) or a full StrokeStyle. Diagonal borders
are diagonal_tl_br and diagonal_bl_tr.
Header:
{
"show": true,
"repeat_on_page_break": true,
"rows": [
{
"cells": [
{ "content": "Product", "col_span": 2 },
{ "content": "Stock", "row_span": 2 }
]
}
],
"cell": {
"fill": { "color": "#F3F4F6" },
"text": { "font_weight": "bold" }
}
}Row header:
{
"cell": {
"fill": { "color": "#F8F8F8" },
"text": { "font_weight": "bold" }
}
}Body:
{
"cell": {},
"alternate_fill": { "color": "#FAFAFA" }
}Rules:
- Grouped column headers go in
header.rows. Leaf headers come fromcolumns[].header. header.rows[].cells[].contentandcolumns[].headeraccept the same value types as body cells.- The top-left corner cell is the leftmost row-header column's own
headervalue. header.show = falseignoresheader.rows,columns[].header, andheader_cell.
{
"top": { "color": "#111111", "width": 0.3 },
"right": { "color": "#111111", "width": 0.3 },
"bottom": { "color": "#111111", "width": 0.3 },
"left": { "color": "#111111", "width": 0.3 },
"horizontal": { "color": "#D1D5DB", "width": 0.2 },
"vertical": false
}Each side accepts false (suppress) or a full StrokeStyle. Double lines
use compound: { "kind": "double", "gap": <mm> } on the stroke.
{
"keep_spans_together": true,
"row_min_height": 10,
"header_min_height": 12
}| Field | Type | Default | Notes |
|---|---|---|---|
keep_spans_together |
boolean |
true |
Required true whenever any cell has row_span > 1. |
row_min_height |
number |
— | Minimum body row height (mm). |
header_min_height |
number |
— | Minimum header row height (mm). |
Width:
columns.length >= 1.- All
columns[].widthusefixed,percent, orauto. - If
table.widthis set:fixedcolumns reserve their mm.percentcolumns take a share oftable.width.autocolumns absorb the remainder, sized by content measurement.- With no
autocolumn, the resolved widths must filltable.widthexactly (otherwiseAPI-002).
- If
table.widthis omitted, all columns must befixed. - The sum of
percentwidths cannot exceed100.
Style precedence (later wins):
settings.defaultstable.cell- Zone-level
header.cell/row_header.cell/body.cell columns[].cell/columns[].header_cell- Per-cell
cell.style
Border precedence (later wins):
gridtable.cell.borders- Zone-level
cell.borders - Column-level
cell.borders/header_cell.borders - Per-cell
cell.style.borders
stack solves the "table followed by trailing content" problem common in
invoices and statements. It composes a table with one or more block sections
that flow under it.
{
"type": "stack",
"gap": 6,
"children": [
{
"type": "table",
"x": 18,
"y": 123,
"width": 180,
"columns": [
{ "key": "description", "header": "Description", "width": { "mode": "auto" } },
{ "key": "amount", "header": "Amount", "width": { "mode": "fixed", "value": 40 },
"cell": { "text": { "text_align": "right" } } }
],
"rows": [
{ "description": "Cloud compute", "amount": "$823.10" },
{ "description": "Replica storage", "amount": "$214.50" },
{ "description": "Cache warmup", "amount": "$306.05" }
]
},
{
"type": "block",
"elements": [
{ "type": "text", "x": 128, "y": 0, "content": "Subtotal" },
{ "type": "text", "x": 168, "y": 0, "content": "$1,343.65" },
{ "type": "line", "x1": 128, "y1": 7, "x2": 178, "y2": 7 }
]
}
]
}Top-level fields:
| Field | Type | Required | Notes |
|---|---|---|---|
gap |
number |
No | Vertical gap (mm) between consecutive children. Default 0. |
children |
StackChild[] |
Yes | Length >= 2. |
Rules:
stackmay only appear directly insidepages[].elements.children[0]must be atable.children[1..]must beblockelements.
Block element:
{
"type": "block",
"elements": [
{ "type": "text", "x": 128, "y": 0, "content": "Subtotal" },
{ "type": "text", "x": 168, "y": 0, "content": "$1,343.65" }
]
}Rules:
blockdoes not acceptx/y/width/height.block.elements[].xkeeps the existing body-element semantics.block.elements[].yis relative to the block's start.blockcannot nesttable,stack, or anotherblock.- Block height is measured from its inner elements.
Pagination:
- The table paginates using its own remainder logic.
- Trailing blocks only start laying out after the table is fully rendered.
- A block that does not fit on the current page moves entirely to the next page. The preceding
gapis dropped on the new page. - A block taller than the available page height returns
API-002.
header and footer are page-global decorations applied to every page.
{
"header": {
"height": 14,
"elements": [
{
"type": "text",
"x": 12,
"y": 8,
"content": "Monthly Report",
"style": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 10,
"font_weight": "bold",
"color": "#111827",
"width": 80
}
}
]
},
"footer": {
"height": 12,
"elements": [
{
"type": "text",
"x": 150,
"y": 6,
"content": "Page 1 / 12",
"style": {
"font_family": "NotoSans-Regular",
"font_mode": "prefer",
"font_size": 8,
"color": "#6B7280",
"width": 40,
"text_align": "right"
}
}
]
}
}| Field | Type | Required | Notes |
|---|---|---|---|
height |
number |
Yes | Region height (mm). |
elements |
Element[] |
No | Region elements. May be empty. |
Coordinate semantics:
header.elementsuse absolute page coordinates. By convention place them insidey ∈ [0, header.height].footer.elementsare auto-shifted bypage.height - footer.heightat render time. Write them as ify = 0is the top of the footer region.- Header height does not push body elements down; body elements still use absolute coordinates.
- Footer height is subtracted from the auto-paginated body region, so body overflow starts a new page above the footer.
Recommended:
- Set
header.heightandfooter.heightclose to actual content height. Oversized regions waste body space. - For per-page-different headers/footers, do not use these global regions; embed the content in
pages[].elementsinstead.
Text inside header / footer follows the section profile (§4.6.4): no
list, no page_break, no frame.overflow = paginate, no multi-column.
layers are document-level decorative layers. They do not participate in
body pagination and do not consume header / footer height. Use them
for diagonal "DRAFT" / "PRIVATE COPY" watermarks, page background colour
or paper textures, and "PAID" / approval stamps.
{
"layers": {
"background": {
"repeat": "all_pages",
"elements": [
{
"type": "rect",
"x": 0, "y": 0,
"width": 215.9, "height": 279.4,
"fill": { "color": "#FFFBEB" }
}
]
},
"watermark": {
"repeat": "all_pages",
"opacity": 0.12,
"template": {
"type": "text",
"content": "PRIVATE COPY"
},
"style": {
"font_family": "NotoSans-Regular",
"font_size": 10.5,
"font_weight": "bold",
"color": "#B91C1C",
"width": 56,
"text_align": "center"
},
"layout": {
"preset": "diagonal_tile",
"angle": 330,
"gap_x": 18, "gap_y": 16,
"offset_x": 8, "offset_y": 10,
"stagger_x": 34
}
},
"stamp": {
"repeat": "last_page",
"elements": [
{
"type": "circle",
"cx": 171, "cy": 228, "r": 18,
"stroke": { "color": "#B91C1C", "width": 0.9 }
},
{
"type": "text",
"x": 152, "y": 220,
"content": "PAID",
"style": {
"font_size": 12,
"font_weight": "bold",
"color": "#B91C1C",
"width": 38,
"text_align": "center"
}
}
]
}
}
}Three layer slots:
| Slot | Render order | Spec shape | Use for |
|---|---|---|---|
background |
Behind body | repeat, elements[] |
Page colour, paper backgrounds, decorative frames. |
watermark |
Above body | Algorithmic spec: template + style + layout + opacity |
Diagonal-tiled "DRAFT" / "PRIVATE COPY" / brand text. |
stamp |
Top-most | repeat, elements[] |
"PAID" / approval marks, often on the last page. |
Common fields:
repeat:all_pages(default),first_page,last_page.
background / stamp element rules:
- Allowed types:
text,image,rect,line,circle,ellipse,polygon,link. - Forbidden types:
table,stack.
watermark rules:
template.typeis currently onlytext.layout.preset:center,tile,diagonal_tile,arc_outside,arc_inside,wave.- A single
watermarkselects onelayout.presetbehavior. Standard placement (center,tile,diagonal_tile) and path text placement (arc_outside,arc_inside,wave) are mutually exclusive. opacityis in[0, 1].- For
center,tile, anddiagonal_tile,layout.angleis text rotation in degrees. - For
arc_outsideandarc_inside, text is placed along a circular baseline. This is path text placement, not glyph-outline distortion.center_x/center_y: optional circle centre in mm. Defaults to page centre.radius: optional circle radius in mm. Defaults tomin(page.width, page.height) * 0.28.angle: optional arc anchor angle in degrees. Defaults to90forarc_outsideand270forarc_inside.
- For
wave, text is placed along a sine-wave baseline.start_x/start_y: optional wave start point in mm. Defaults to centred horizontally and page centre vertically.amplitude: optional wave amplitude in mm. Default6.wavelength: optional wave length in mm. Default42.
- Path watermark limits are enforced by validation:
- Single-line text only.
- At most 32 grapheme clusters.
style.width,style.height,style.text_overflow,style.shrink_to_fit,style.background,style.decoration,style.link_style, andstyle.wrap_policyare not supported for path watermarks.gap_x,gap_y, andstagger_xare only valid for tiled presets.
Path watermark example:
{
"template": {
"type": "text",
"content": "PRIVATE COPY"
},
"style": {
"font_size": 16,
"font_weight": "bold",
"color": "#B91C1C"
},
"opacity": 0.14,
"layout": {
"preset": "arc_outside",
"center_x": 105,
"center_y": 148,
"radius": 58,
"angle": 90
}
}{
"settings": {
"defaults": {
"text": { "font_family": "NotoSans-Regular", "font_size": 11, "color": "#111111" },
"stroke": { "color": "#000000", "width": 0.4 },
"fill": { "color": "#FFFFFF", "opacity": 1.0 },
"shape": { "corner_radius": 0 }
},
"metadata": {
"title": "gPdf Document",
"author": "Acme Cloud Inc."
},
"output": {
"mode": "file",
"file_name": "invoice-20260310.pdf"
},
"profile": "pdfa-2b",
"page_margin": { "top": 10, "right": 12, "bottom": 10, "left": 12 }
}
}| Field | Type | Notes |
|---|---|---|
defaults |
Defaults |
Global default styles. See §4.14.1. |
metadata |
Metadata |
PDF metadata. See §4.14.2. |
output |
OutputSettings |
Response shape. See §4.14.3. |
profile |
string |
PDF/A profile. See §6.4. |
page_margin |
PageMargin |
Global margin. See §4.3.2. |
e_invoice |
EInvoiceSettings |
Only valid on POST /api/v1/e-invoice/render. Sending it to JSON Render returns API-002. See §5. |
security |
SecuritySettings |
PDF password + permission protection. See §4.15. |
{
"defaults": {
"text": { "font_family": "NotoSans-Regular", "font_size": 11, "color": "#111111" },
"stroke": { "color": "#000000", "width": 0.4 },
"fill": { "color": "#FFFFFF", "opacity": 1.0 },
"shape": { "corner_radius": 0 }
}
}| Subkey | Type | Notes |
|---|---|---|
text |
TextStyle |
Default text style. If font_family is set here without font_mode, strict is used. Use font_mode = "prefer" with the same font_family when this default should allow fallback, for example mixed Latin + CJK text. |
stroke |
StrokeStyle |
Default stroke for shapes and table grids. |
fill |
FillStyle |
Default fill. Default opacity is 0 (transparent) when omitted. |
shape |
ShapeDefaults |
corner_radius (mm) for default rounded rectangles. |
settings.defaults only accepts the four nested groups above. The legacy
flat fields font_family, font_size, color, unit are rejected with
API-002.
{
"metadata": {
"title": "Monthly Report",
"author": "Acme Cloud Inc.",
"subject": "Operations digest",
"creator": "OpsBot 4.2",
"producer": "gPdf",
"language": "en"
}
}| Field | Type | Notes |
|---|---|---|
title |
string |
|
author |
string |
|
subject |
string |
|
creator |
string |
The application that created the source content. |
producer |
string |
The renderer. |
language |
string |
BCP-47 language tag (e.g. en, de, zh-Hans). |
Normalisation:
- Empty or whitespace-only strings are treated as not provided.
- A token's policy may strip metadata fields it has no permission for; the request still succeeds.
- Stripped fields fall back to the policy's
default_metadata. title,creator,producer,languageuse system fallbacks when no value or default exists.authorandsubjectremain empty if no value or default exists.
{
"output": {
"mode": "file",
"file_name": "invoice-20260310.pdf"
}
}| Field | Type | Default | Notes |
|---|---|---|---|
mode |
"binary" | "file" |
binary |
Both return the same PDF bytes. Differs in Content-Disposition. |
file_name |
string |
Auto-generated gPdf-MMDDHHmmssSSS.pdf |
Sanitised; .pdf is appended automatically. |
Behaviour:
binary(or omitted):Content-Disposition: inline; filename="...". Browsers preview inline.file:Content-Disposition: attachment; filename="...". Browsers download.- Any value other than
binary/filereturnsAPI-002.
settings.security turns on PDF standard-security-handler encryption
on the output PDF — AES-128 or AES-256, optional open password, optional
owner password, and eight per-action permission flags (print / copy /
modify / annotate / fill_forms / extract_accessibility / assemble /
print_high_quality). Only valid on POST /api/v1/pdf/render. Mutually
exclusive with settings.profile (PDF/A) and settings.e_invoice —
combining either returns API-002.
{
"settings": {
"security": {
"algorithm": "aes_128",
"open_password": "reader-demo",
"owner_password": "owner-demo",
"permissions": {
"print": true,
"modify": false,
"copy": false,
"annotate": false,
"fill_forms": true,
"extract_accessibility": true,
"assemble": false,
"print_high_quality": true
}
}
}
}The example uses
aes_128to match the default and the lowest tier that ships encryption (Pro). For AES-256 — PDF 2.0 standard-security-handler revision 6 — switchalgorithmtoaes_256; that path is Enterprise-only (see tier policy below). Both algorithms accept the same field shape.
| Field | Type | Notes |
|---|---|---|
algorithm |
"aes_128" | "aes_256" |
Default aes_128. AES-128 = standard-security-handler revision 4 (R=4, V=4, AESV2 crypt filter, PDF 1.7 output). AES-256 = revision 6 (R=6, V=5, AESV3 crypt filter, PDF 2.0 output). |
open_password |
string |
Password required to open the PDF. Omitted, null, empty or whitespace-only = encryption disabled. Max 32 UTF-8 bytes. |
owner_password |
string |
Owner password granting full rights regardless of permissions. Must differ from open_password. Max 32 UTF-8 bytes. Enterprise policy. |
permissions |
Permissions |
8 booleans, default true. Restricting any flag requires owner_password. Enterprise policy. |
Permissions (each defaults to true):
| Flag | Effect when false |
|---|---|
print |
Printing is blocked. |
print_high_quality |
High-quality printing is blocked (a low-res render may still be allowed by print). |
modify |
Content edits other than annotations / form-field fill are blocked. |
copy |
Selecting and copying text / graphics is blocked. |
annotate |
Adding or modifying annotations AND form-field definitions is blocked. |
fill_forms |
Filling existing form fields is blocked (independent of annotate). |
extract_accessibility |
Extracting text / graphics for accessibility tools is blocked. |
assemble |
Inserting / rotating / deleting pages, and creating bookmarks / thumbnails, is blocked. |
Tier policy:
| Pro | Enterprise | |
|---|---|---|
| Algorithms | AES-128 only | AES-128 or AES-256 |
open_password |
✓ | ✓ |
owner_password |
— | ✓ |
permissions |
— | ✓ |
Behaviour:
- The metadata stream is encrypted alongside the rest of the document;
/EncryptMetadatawritestrue. - The token's xAdmin PDF Policy must allow
document.security.allow = true, anddocument.security.allowed_algorithmsmust contain the requested algorithm. - A
settings.securityobject that contains no validopen_password, noowner_password, and no restrictedpermissionsis a no-op — the request renders an unencrypted PDF and does NOT enter the encryption write path. - Unrecognised
algorithmvalues are rejected at JSON deserialisation withAPI-001(settings.security.algorithm must be aes_128 or aes_256). - All other
settings.securitymisuses (password >32 UTF-8 bytes, policy disallow, combination withprofile/e_invoice, semantic violation) returnAPI-002with the offending field inmessage.
When a field is omitted from an element, gPdf walks this chain to fill it in:
- The element's own field (e.g.
line.stroke.width). settings.defaults(e.g.defaults.stroke.width).- Service-level defaults.
Font-family inheritance is not "fall back to auto when missing". It is "keep walking up the chain until a font is declared, and only enter auto mode if the entire chain is silent". Practical consequences:
- An element with
font_familyset continues to use that family even if children omit it. - Children inherit the explicit family until one explicitly overrides.
- Auto mode only activates when no ancestor has set a family at all.
The e-invoice family — Factur-X / ZUGFeRD packaging, capabilities, render, async strict-validation jobs, and artifact download — has its own dedicated reference at /docs/e-invoice-api/.
It moved out of this document on 2026-05-09 to give the e-invoice flows
(inline_pdf vs object, basic vs strict validation, residency
profiles) the room they need without bloating the JSON Render reference.
All gPdf errors share the JSON envelope shown in §1.1.
Codes fall into four families: client (API-0xx), authentication (API-1xx),
billing/entitlement (API-2xx), and rendering/system (API-5xx / API-9xx).
| Code | HTTP | Family | Trigger | Typical message |
What to do |
|---|---|---|---|---|---|
API-001 |
400 |
Client | Body is not valid JSON. | Invalid JSON payload |
Validate with jq . or a JSON linter before sending. |
API-002 |
400 |
Client | Body parsed but failed schema or business validation. | <field> must be >= 0, Missing required field <name>, Field type mismatch |
Read message. The field name is always included. |
API-004 |
400 |
Client | Total page count exceeds the per-request limit (token policy or platform max, whichever is smaller). | page count exceeds max_pages_per_request |
Split the request into smaller batches, or request a higher policy limit. |
API-007 |
400 |
Client | Embedded image bytes exceed the per-image limit set by the active token policy. | image bytes exceeds max_image_bytes |
Re-encode the image at a smaller size, or use source.kind = "asset" to reference a pre-uploaded asset. |
API-008 |
413 |
Client | Request body exceeds the platform body limit (default 16 MiB; some deployments differ). |
Request body too large |
Reduce inline payload (especially base64 images). Consider splitting the document. |
API-101 |
401 |
Auth | Authorization header is missing or not in Bearer <token> form. |
Missing or malformed Authorization header |
Add the header. The format is exactly Bearer followed by the token. |
API-102 |
401 |
Auth | Authentication failed (unknown token, environment mismatch, signature failure). | Authentication failed (redacted) |
Verify the token belongs to the environment you are calling. |
API-103 |
401 |
Auth | Token is blacklisted (revoked, suspended, or otherwise invalidated). | Authentication failed (redacted) |
Rotate the token via the Console, or contact support. |
API-201 |
402 |
Billing | No active subscription entitlement for this token. | No active subscription |
Activate or renew a plan in the Console. |
API-202 |
402 |
Billing | Subscription expired. | Subscription expired |
Renew in the Console. |
API-203 |
402 |
Billing | Subscription quota exceeded. | Quota exceeded |
Wait for the next billing cycle, top up, or upgrade the plan. |
API-204 |
402 |
Billing | Wallet balance insufficient for overage. | Insufficient wallet balance |
Top up via the Console. |
API-501 |
500 |
Render | PDF generation failed during rendering. | Detailed message describing the cause | Inspect message. Often points to invalid font, asset, or coordinate. |
API-502 |
500 |
Render | PDF/A compliance check failed after rendering. | PDF/A compliance check failed: <reason> |
Adjust the offending field (commonly fonts not in the embedded set, or non-PDF/A images). |
API-503 to API-507 |
500 |
Render | Specific rendering subsystem failures (font, asset resolution, layout). | Detailed message | Inspect message. These preserve actionable detail intentionally. |
API-504 |
500 |
Render | Resource loading failed. For fonts, this includes auto / prefer fallback exhaustion. |
Font fallback failed for: <run> or another resource message |
Provide a font_family that covers the script, upload the missing font as an asset, or verify the referenced image/font asset. |
API-900 |
500 |
System | Internal system error. | Redacted message | Retry once. If it persists, contact support with the req_id. |
API-999 |
500 |
System | Unknown internal error. | Redacted message | Same as API-900. |
Notes on validation errors (API-002):
API-002is the most common error. Common triggers include:xandx_anchorprovided on the same element (mutually exclusive).- Custom
page.width/page.heightoutside the supported10 <= value <= 2000 mmrange. font_modeprovided without a same-levelfont_family.- Explicit font in
strictmode that does not cover the submitted text. - Invalid
link(unsupported URL scheme, page index out of bounds, malformedpadding/border). - Invalid
table(unknown column key,table.widthcannot allocate a positive width to undeclared columns, invalid span). - Invalid
profilevalue. - Invalid
settings.security(password >32 UTF-8 bytes, policy doesn't permit algorithm, combined withsettings.profileorsettings.e_invoice, orpermissionsprovided withoutowner_password). Unrecognisedalgorithmvalue is rejected at JSON deserialisation withAPI-001.
Notes on redaction:
API-102,API-103, andAPI-9xxdeliberately use generic messages. They will not tell you whether a token exists or why authentication failed. This is by design — to defeat token enumeration.API-201toAPI-204(billing) preserve actionable text so end users know whether to renew, top up, or wait for the cycle.API-501toAPI-507(render) preserve actionable text so engineering can debug.
Three kinds of limits apply to every request:
- Platform limits — fixed across all tenants, set per environment.
- Policy limits — bound to the token, set when the plan is provisioned.
- Request-shape limits — encoded in the request schema itself.
| Limit | Default | Scope | Override path | Triggered error |
|---|---|---|---|---|
| Request body size | 16 MiB |
Platform | Per-deployment env var; some private deployments raise it. | API-008 |
| Pages per request | No platform default — entirely policy-driven | Token policy | Plan or per-token policy in the Console. | API-004 |
| Image bytes per element | No platform default — only enforced if the policy sets max_image_bytes |
Token policy | Plan or per-token policy in the Console. | API-007 |
Template batch size (data array) |
10 items |
Platform | Not configurable. Split into multiple requests if you need more. | API-002 (Template render data item count ... exceeds max 10) |
| URL TTL for e-invoice artifacts | 900 seconds |
Request | delivery.url_ttl_seconds, range 1..900 |
API-002 if out of range |
| Retention for e-invoice artifacts | 23 hours |
Request | retention.ttl_hours, range 1..23 |
API-002 if out of range |
xml.content (e-invoice) |
2 MiB |
Platform | Not configurable. | API-002 |
How to pre-check on the client side:
- Sum your request body bytes and reject above ~
14 MiBbefore sending. This leaves headroom for proxies and avoidsAPI-008. - If you embed images via
source.kind = "base64", the base64 form is ~33% larger than the raw bytes. A 5 MiB raw JPEG becomes ~6.7 MiB in JSON. - If you batch templates, never put more than
10items indata. Use multiple HTTP calls for larger batches.
| Header | Direction | Always present | Notes |
|---|---|---|---|
Content-Type |
Response | Yes | application/pdf on success, application/json on error. E-invoice job and artifact endpoints also return application/json. |
Content-Disposition |
Response | Yes (PDF responses only) | inline; filename="..." by default; attachment; filename="..." when output.mode = "file". |
X-Request-Id |
Both | Yes | Echoed from the request if the client supplied it; otherwise generated. |
Headers gPdf does not currently emit (and the absence is intentional in v1):
X-RateLimit-Limit,X-RateLimit-Remaining,X-RateLimit-ResetRetry-AfterIdempotency-KeyechoX-Render-Time-MsX-Render-Engine-Version
Do not depend on these. If a future version adds any of them, this document will list them and clients can opt in.
Font resolution is controlled by font_family and font_mode:
- Auto: when no
font_familyis declared anywhere in the inheritance chain, the renderer chooses fonts that cover each text run from the bundled font set.font_mode = "auto"is not a public input value. prefer: whenfont_familyandfont_mode = "prefer"are declared in the same style object, the renderer tries the declared family first and falls back through bundled families for glyphs it cannot cover.strict: whenfont_familyis declared withfont_mode = "strict", or declared withoutfont_mode, the renderer must use that family. If the family cannot cover the text, the request fails withAPI-002.
Practical CJK guidance:
- If you do not declare
font_familyanywhere, auto mode can select bundled CJK fonts. - If you declare a Latin/default family such as
NotoSans-RegularorRobotoMono-Regularand the text may contain Chinese, Japanese, or Korean characters, setfont_mode = "prefer"in the same style object. - If you declare a family without
font_mode = "prefer", the request is strict. Mixed CJK text that the declared family cannot cover returnsAPI-002.
Failure modes:
- Strict coverage miss →
API-002. Adjust the text or pick a font that covers it, or switch the same style object tofont_mode = "prefer". - Auto / prefer total fallback miss →
API-504. The bundled set could not cover the text in any family.
Bundled CJK fallback maps Hangul to NotoSansKR-Regular, Japanese kana to
NotoSansJP-Regular, and CJK ideographs / fullwidth punctuation to
NotoSansSC-Regular. The bundled set also includes Latin, Greek, Cyrillic,
Arabic, Hebrew, Bengali, Tamil, Thai, Vietnamese, Osage, and monospace fonts.
This is not full Unicode coverage; unsupported scripts or symbols may still
fail in auto / prefer mode with API-504. Custom fonts can be uploaded as
assets via the Console and referenced by font_family.
PDF/A profiles:
| Profile | Use case |
|---|---|
pdfa-1b |
PDF/A-1b. Long-term archival, oldest baseline. |
pdfa-2b |
PDF/A-2b. Common archival profile. |
pdfa-3b |
PDF/A-3b. Required for embedded XML (e-invoice). |
pdfa-4 |
PDF/A-4. Newer archival profile. |
pdfa-2u |
PDF/A-2u. Unicode-mapped variant of 2b. |
pdfa-3u |
PDF/A-3u. Unicode-mapped variant of 3b. |
pdfa-ua1 |
PDF/UA-1. Accessibility profile. |
Set the profile via settings.profile. The chosen profile gates which
features can be used (e.g. transparency, embedded files). Violations return
API-502.
- Length unit: millimetre (mm). Floating-point values are accepted; the renderer rounds to PDF user-space at output.
- Origin: top-left of the page, or the content box if
page_marginis set. - X axis: rightward. Y axis: downward.
rotationis in degrees, clockwise. Most elements accept only0/90/180/270.textandimageaccept any integer angle. Barcodes and their attachedbarcode_textaccept only the four cardinal angles.
This document tracks the public API contract. Internal changes that do not affect callers are not listed here.
- Added constrained path-text watermark presets to
layers.watermark.layout.preset:arc_outside,arc_inside, andwave. The feature places short single-line text along circular or sine-wave baselines; it does not distort glyph outlines. Validation caps path watermark text at 32 grapheme clusters and rejects layout/style fields that would make rendering expensive or ambiguous.
- Docs-only outline reshuffle (no API changes). §4.13 renamed
Layers→Layers (background, watermark, stamp)so the watermark feature is visible in the right-hand page outline. §4.14.4Security (password + permissions)promoted to its own H3 §4.15Security: PDF password and document permissionsfor the same reason. Old §4.15Default value precedencerenumbered to §4.16. Anchor URLs for §4.13, §4.15, and §4.16 changed; the §4.15 anchor that pointed at default-value-precedence now points at the security section, so update any external bookmarks accordingly. All field names (settings.security,layers.watermark, etc.) and behaviour are unchanged.
- Added
settings.securityto JSON Render (POST /api/v1/pdf/render). Optional AES-128 or AES-256 PDF encryption withopen_password,owner_password(Enterprise policy), and 8 permission bits (print,modify,copy,annotate,fill_forms,extract_accessibility,assemble,print_high_quality). Passwords capped at 32 UTF-8 bytes. Mutually exclusive withsettings.profile(PDF/A) andsettings.e_invoice— both combinations returnAPI-002. Full field table, per-bit effect table, and tier matrix at §4.15 (was §4.14.4 at ship-time; renumbered 2026-05-17). - Added Security section to the request reference (originally §4.14.4, now §4.15).
- §6.1
API-002trigger list extended withsettings.securitymisuse cases (password >32 UTF-8 bytes, policy doesn't permit algorithm, combined withsettings.profile/settings.e_invoice, orpermissionsprovided withoutowner_password). Unrecognisedalgorithmvalue returnsAPI-001at JSON deserialisation, notAPI-002.
- First publication of the consolidated public API reference.
- Documented the e-invoice job and artifact endpoints (
GET /api/v1/e-invoice/jobs/{job_id}and.../artifacts/{artifact}). - Single consolidated error-code reference at §6.1.