decode-gs1-ai.
When to use
- A consumer-facing surface needs a QR code or URL for a product: packaging, shelf-talker, email receipt, branded resolver page.
- An ERP / PIM / TMS system has structured product+lot+serial data and needs to emit GS1 Digital Link strings to label printers or partner feeds.
- An agent received a raw GTIN and wants to construct the canonical Closient resolver URL to hand back to a user.
decode-gs1-ai.
Anatomy of a Digital Link
| Component | Rules |
|---|---|
| Host | Any HTTPS host that supports the resolver protocol. id.gs1.org is the canonical GS1-managed host; you may use your own domain (e.g. id.closient.com, acme.gtin.one). |
| Primary identifier | Exactly one of /01/<gtin>, /00/<sscc>, /253/<gdti>, /255/<gcn>, /401/<ginc>, /402/<gsin>, /414/<gln>, /417/<gln>, /8003/<grai>, /8004/<giai>, /8006/<gctin>, /8010/<cpid>, /8013/<gmn>, /8017/<gsrn>, /8018/<gsrn>. |
| Qualifiers | Subsequent /{AI}/{value} segments that narrow the primary (e.g. /10/<lot>, /17/<expiry>, /21/<serial>). |
| Data attributes | Long-tail AIs go in the query string: ?15=271231 for AI 15 best-before. |
linkType= | Tells the resolver what kind of destination to serve (PIP, product page, instructions, recall, etc.). |
Step 1 — Validate inputs
Before composing anything, normalize:Step 2 — Compose the primary AI
GTIN is the common case:| Primary identifier | Path |
|---|---|
| GTIN-14 | /01/<gtin14> |
| SSCC | /00/<sscc18> |
| GRAI (returnable asset) | /8003/<grai> |
| GIAI (individual asset) | /8004/<giai> |
| GLN (location) | /414/<gln> |
| GSIN (shipment) | /402/<gsin17> |
| GINC (consignment) | /401/<ginc> |
Step 3 — Append qualifier AIs (canonical order)
GS1 DL §4.4 fixes the order qualifier AIs must appear in: batch/lot (10) before expiry (17) before serial (21) when all three are present on a GTIN-anchored URL. The canonical order:Step 4 — Encode special characters in values
GS1 Digital Link values are URL path segments, so anything outside the unreserved set (A-Z a-z 0-9 - . _ ~) must be percent-encoded:
%20), + (%2B), & (%26), / (%2F), ?
(%3F), # (%23). Don’t pre-escape with \\ — that’s a different
encoding.
Step 5 — Attach data-attribute AIs (query string)
AIs that aren’t qualifiers go in the query string as<ai>=<value>:
| AI | What it means | Example query param |
|---|---|---|
| 15 | Best-before date YYMMDD | 15=271231 |
| 16 | Sell-by date YYMMDD | 16=271231 |
| 11 | Production date YYMMDD | 11=270101 |
| 8005 | Price per UoM | 8005=125000 |
| 3922 | Price (currency-implied) | 3922=12.50 |
Step 6 — (Optional) attach linkType to drive resolution intent
A Digital Link URL can point at one identifier but many
destinations: product info page, recall notice, instructions, video,
loyalty enrollment. The linkType query string parameter tells the
resolver which destination the agent (or end user) wants:
https://www.gs1.org/voc/). Closient’s resolver maps these to
hosted-page destinations and the Resolver model’s
destination_type column — see apps/resolver/gs1_link_types.py for
the exhaustive list this codebase recognizes.
If you don’t set linkType, the resolver returns its default
destination — typically the brand’s PIP or the public Closient
product page.
Step 7 — Pick the host
| Host | Use it when |
|---|---|
https://id.gs1.org | You want GS1’s official global resolver to handle the URL. |
https://id.closient.com | You’re publishing through Closient and want our resolver to serve hosted pages, redirects, and analytics. |
https://<brand>.gtin.one | Closient-managed branded resolver subdomain for a specific brand. |
https://<brand>.example.com | The brand’s own CNAME pointing at Closient. |
id.closient.com and *.gtin.one; the Brand.preferred_resolver_host
column stores which one to emit for that brand’s products.
Full example
Inputs:- GTIN-14:
09506000134352 - Lot:
ABC123 - Expiry:
2030-09-01(=300901YYMMDD) - Serial:
SN9999 - Best-before:
2027-12-31(=271231YYMMDD, AI 15 → query string) - Link type: PIP (product information page)
- Host:
https://id.closient.com
Common mistakes
- Wrong qualifier order. Emitting
/01/<gtin>/21/<serial>/10/<lot>is parseable but not canonical; the resolver redirects to the canonical ordering and analytics will conflate the two URLs. - Leaving out the qualifier on a serialized item. A GTIN+serial URL
(
/01/<gtin>/21/<serial>) resolves to that specific item; a GTIN-only URL (/01/<gtin>) resolves to the class. Emit the qualifier when you have the data — class-level resolution can’t drive item-level experiences like recalls. - Encoding the AI itself. AI digits are path literals and must NOT be percent-encoded — only the value slot is.
- Wrong link-type CURIE.
linkType=gs1:pip(correct CURIE) versuslinkType=pip(bare token, ambiguous) — always emit the namespaced form. Closient’s test suite normalizes both, but partner resolvers may not. - Hardcoding
id.gs1.orgfor brand-owned products. Closient brands have their own preferred resolver host; use that so the brand owns the surface and gets the analytics. - Not URL-encoding non-ASCII or reserved characters in lot/serial.
A lot of
LOT-2024/Q1becomesLOT-2024%2FQ1.
What Closient does with the URL
Once emitted, the URL can be:- Encoded as a QR code — Closient’s QR rendering service
(
apps/products/services/qr_url.py) takes any Digital Link URL and produces a print-ready QR. See thePrint Digital LinkMCP tool orPOST /products/api/v1/qr/render. - Hosted by Closient’s resolver — if the URL points at
id.closient.comor a*.gtin.onebrand domain, Closient serves the destination (hosted page, redirect, JSON linkset, branded video, etc.) based onlinkTypeand the brand’s resolver configuration. - Verified for serialized items — call
GET /resolver/api/v1/verify/01/{gtin}/21/{serial}to confirm an item is authentic + active.
Related skills
decode-gs1-ai— inverse: take a Digital Link URL apart into AIs.resolve-gtin— once you have the URL, fetch product detail.capture-epcis-event— emit a supply-chain event keyed on the same GTIN + serial.closient-api-quickstart— authenticate before calling/products/api/v1/qr/renderor/resolver/api/v1/verify/....
Authoritative references
- GS1 Digital Link Standard 1.4 — sections 4 (URI syntax), 5 (canonicalization), 6 (link types).
- ISO/IEC 18975:2024 — formalization of GS1 Digital Link.
- GS1 link-type vocabulary:
https://www.gs1.org/voc/. - Closient internal:
apps/resolver/digital_link_builder.py,apps/resolver/gs1_link_types.py,apps/products/services/qr_url.py.