diff --git a/.gitignore b/.gitignore index d72a9634..1992ebaa 100644 --- a/.gitignore +++ b/.gitignore @@ -35,6 +35,7 @@ yarn-error.log testem.log /typings .env +minio_data/* # Yarn .yarn/* diff --git a/app-config.acrs.json b/app-config.acrs.json new file mode 100644 index 00000000..7e5333a9 --- /dev/null +++ b/app-config.acrs.json @@ -0,0 +1,2865 @@ +{ + "name": "ACRS", + "styles": { + "primaryColor": "rgb(35, 138, 186)", + "secondaryColor": "black", + "tertiaryColor": "black" + }, + "generalFeatures": [ + { + "name": "General features", + "type": "", + "styles": { + "primaryColor": "rgb(35, 138, 186)", + "secondaryColor": "black", + "tertiaryColor": "black" + }, + "features": [] + } + ], + "apps": [ + { + "name": "Top Line Steel", + "type": "producer", + "assets": { + "logo": "https://storage.googleapis.com/acrs-assets/logos/Top-Line-Steel-Logo.jpg", + "brandTitle": "Top Line Steel", + "passportVC": "", + "transactionEventVC": "" + }, + "styles": { + "primaryColor": "#b22166", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "productIdentifier": [ + { + "scheme": "https://identifier.example.org/steel", + "identifierValue": "0109359502000010", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/steel/12345/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://batch.example.org", + "identifierValue": "BATCH-67890", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://batch.example.org/67890/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://item.example.org", + "identifierValue": "ITEM-112233", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://item.example.org/112233/binding" + } + } + ], + "classification": [ + { + "scheme": "https://unstats.un.org/unsd/classifications/Econ/cpc", + "classifierValue": "41231", + "classifierName": "Steel Rods", + "classifierURL": "https://vocabulary.uncefact.org/unlocode#AUSYD" + } + ], + "modelName": "Steel Model X", + "image": "", + "description": "High-quality steel rods for construction.", + "furtherInformation": "https://example.org/products/steel_model_x", + "manufacturedDate": "2024-05-20", + "dimension": { + "weight": { "value": 1500, "unit": "KG" }, + "length": { "value": 6, "unit": "MTR" }, + "width": { "value": 0.1, "unit": "MTR" }, + "height": { "value": 0.1, "unit": "MTR" }, + "volume": { "value": 0.6, "unit": "CMT" } + }, + "characteristic": {}, + "manufacturer": { + "id": "did:example:123456789abcdefghi", + "name": "Steel Manufacturer Inc.", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-56789", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/56789/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-1234", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/1234/binding" + } + } + ], + "name": "Main Steel Plant", + "location": "https://plus.codes/4RRG6RJX+W3", + "operatedBy": { + "id": "did:example:567890abcdefghi", + "name": "Steel Manufacturer Inc.", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-56789", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/56789/binding" + } + } + ] + } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://textileexchange.org/materials", + "classifierValue": "STEEL", + "classifierName": "Carbon Steel" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "conformityClaim": [ + { + "topic": "Structural Steel Certification", + "standardOrRegulation": "https://example.org/standards/environment", + "criteriaReference": "https://example.org/standards/environment/criteria", + "claimedValues": [], + "benchmarkValue": {}, + "benchmarkReference": "https://example.org/benchmarks/environment", + "conformance": true, + "conformityEvidence": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://acrs.pyx.io/verify?q=%7B%22payload%22%3A%7B%22uri%22%3A%22https%3A%2F%2Fstorage.googleapis.com%2Fverifiable-credentials%2Fconformity-credentials%2Ftop-line-steel-dcc.json%22%7D%7D" + } + } + ], + "recyclingInstruction": "https://example.org/recycling/steel_model_x" + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Steel

Business identifier
85 652 929 014
Location
Rooty Hill, NSW
Identity verification
85 652 929 014
Other evidence

Steel Compliance Certificate

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Manuf. Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Length

{{credentialSubject.dimension.length.value}}{{credentialSubject.dimension.length.unit}}

Width

{{credentialSubject.dimension.width.value}}{{credentialSubject.dimension.width.unit}}

Height

{{credentialSubject.dimension.height.value}}{{credentialSubject.dimension.height.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Steel Passport", + "dlrIdentificationKeyNamespace": "gs1", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/productIdentifier/0/identifierValue" + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "topLineSteel_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Sale Steel", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "topLineSteel_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:api.acrs.pyx.io:top-line-steel", + "name": "Top Line Steel", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:api.acrs.pyx.io:steel-processor", + "name": "Steel Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyNamespace": "gs1", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "topLineSteel_dpps", "keyPath": "/epcList/index/name" } + } + ] + } + ] + } + ] + }, + { + "name": "Quality Steel", + "type": "producer", + "assets": { + "logo": "https://storage.googleapis.com/acrs-assets/logos/Quality-Steel-Logo.png", + "brandTitle": "Quality Steel", + "passportVC": "", + "transactionEventVC": "" + }, + "styles": { + "primaryColor": "#000000", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "productIdentifier": [ + { + "scheme": "https://identifier.example.org/steel", + "identifierValue": "0109359502000041", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/steel/67890/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://batch.example.org", + "identifierValue": "BATCH-54321", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://batch.example.org/54321/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://item.example.org", + "identifierValue": "ITEM-445566", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://item.example.org/445566/binding" + } + } + ], + "classification": [ + { + "scheme": "https://unstats.un.org/unsd/classifications/Econ/cpc", + "classifierValue": "41232", + "classifierName": "Steel Coil", + "classifierURL": "https://vocabulary.uncefact.org/unlocode#GBLON" + } + ], + "modelName": "Steel Model Y", + "image": "", + "description": "Steel Coil for construction and infrastructure projects.", + "furtherInformation": "https://example.org/products/steel_model_y", + "manufacturedDate": "2024-06-01", + "dimension": { + "weight": { "value": 2500, "unit": "KG" }, + "length": { "value": 100, "unit": "MTR" }, + "width": { "value": 1, "unit": "MTR" }, + "height": { "value": 0.005, "unit": "MTR" }, + "volume": { "value": 1.8, "unit": "CMT" } + }, + "characteristic": {}, + "manufacturer": { + "id": "did:example:abcdef1234567890", + "name": "Global Steel Co.", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-6789", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/6789/binding" + } + } + ], + "name": "Advanced Steel Plant", + "location": "https://plus.codes/4RRG4WWR+W5", + "operatedBy": { + "id": "did:example:0987654321abcdef", + "name": "Global Steel Co.", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://textileexchange.org/materials", + "classifierValue": "STEEL", + "classifierName": "Alloy Steel" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "conformityClaim": [ + { + "topic": "Structural Steel Certification", + "standardOrRegulation": "https://example.org/standards/environment", + "criteriaReference": "https://example.org/standards/environment/criteria", + "claimedValues": [], + "benchmarkValue": {}, + "benchmarkReference": "https://example.org/benchmarks/environment", + "conformance": true, + "conformityEvidence": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://acrs.pyx.io/verify?q=%7B%22payload%22%3A%7B%22uri%22%3A%22https%3A%2F%2Fstorage.googleapis.com%2Fverifiable-credentials%2Fconformity-credentials%2Fquality-steel-dcc.json%22%7D%7D" + } + } + ], + "recyclingInstruction": "https://example.org/recycling/steel_model_y" + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Steel

Business identifier
93 669 605 341
Location
Smithfield, NSW
Identity verification
93 669 605 341
Other evidence

Steel Compliance Certificate

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Manuf. Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Length

{{credentialSubject.dimension.length.value}}{{credentialSubject.dimension.length.unit}}

Width

{{credentialSubject.dimension.width.value}}{{credentialSubject.dimension.width.unit}}

Height

{{credentialSubject.dimension.height.value}}{{credentialSubject.dimension.height.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Steel Passport", + "dlrIdentificationKeyNamespace": "gs1", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/productIdentifier/0/identifierValue" + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "qualitySteel_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Sale Steel", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "qualitySteel_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:api.acrs.pyx.io:quality-steel", + "name": "Quality Steel", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:api.acrs.pyx.io:steel-processor", + "name": "Steel Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyNamespace": "gs1", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "qualitySteel_dpps", "keyPath": "/epcList/index/name" } + } + ] + } + ] + } + ] + }, + { + "name": "Steel Processor", + "type": "processor", + "assets": { + "logo": "https://storage.googleapis.com/acrs-assets/logos/Steel-Processor-LogoV1.png", + "brandTitle": "Steel Processor", + "passportVC": "", + "transactionEventVC": "" + }, + "styles": { + "primaryColor": "#8f2629", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Process Steel", + "id": "process_product", + "components": [ + { + "name": "RenderCheckList", + "type": "EntryData", + "props": { + "requiredFieldPath": "/vc/credentialSubject/eventID", + "checkBoxLabel": "Imported valid json:", + "nestedComponents": [ + { + "name": "ImportButton", + "type": "EntryData", + "props": { + "label": "Import JSON", + "style": { "margin": "40px auto", "paddingTop": "40px", "width": "80%" } + } + } + ], + "style": { "margin": "40px auto", "paddingTop": "40px", "width": "80%" } + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + }, + { + "name": "BarcodeGenerator", + "type": "Result", + "props": { + "dataPath": "/credentialSubject/outputEPCList/index/name" + } + } + ], + "services": [ + { + "name": "processTransformationEvent", + "parameters": [ + { + "epcisTransformationEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transformation-event-ld.json"], + "renderTemplate": [ + { + "template": " Transformation Event

TRACEABILITY EVENT

Transformation

EVENT ISSUED BY

{{issuer.name}}

Industry
Steel
Business identifier
55 126 272 964
Identity verification
55 126 272 964
Other evidence

Event description

Event ID
{{credentialSubject.eventID}}
Event type
{{credentialSubject.action}}
Description
Assemble Pile Cage
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
In Transit
Business step
Receiving
Process type
Assembly

Transformation

{{#each credentialSubject.outputEPCList}}

{{name}}

OUTPUT

Pile Cage

3M Pile Cage

Sustainability 100%
Confidence 100%
View
{{/each}}

Transformed

{{credentialSubject.inputEPCList.0.name}}

INPUT

Steel Rods

Mfg By: Top Line Steel

Sustainability 100%
Confidence 100%
View

{{credentialSubject.inputEPCList.1.name}}

INPUT

Steel Coil

Mig By: Quality Steel

Sustainability 100%
Confidence 100%
View
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransformationEventCredential"], + "dlrIdentificationKeyType": "gtin", + "dlrLinkTitle": "EPCIS transformation event VC", + "dlrIdentificationKeyNamespace": "gs1", + "dlrVerificationPage": "http://localhost:3003/verify", + "dlrQualifierPath": "" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Steel

Business identifier
55 126 272 964
Identity verification
55 126 272 964
Other evidence

Steel Compliance Certificate

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Traceability

Traceability events specify the “what, when, where, why and how” of the products and facilities that constitute a value chain.

{{#each credentialSubject.traceabilityInformation}}
{{eventType}}View
{{/each}}

Product information

Manuf. Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Length

{{credentialSubject.dimension.length.value}}{{credentialSubject.dimension.length.unit}}

Width

{{credentialSubject.dimension.width.value}}{{credentialSubject.dimension.width.unit}}

Height

{{credentialSubject.dimension.height.value}}{{credentialSubject.dimension.height.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["DigitalProductPassport"], + "dlrIdentificationKeyType": "gtin", + "dlrLinkTitle": "Digital Product Passport", + "dlrIdentificationKeyNamespace": "gs1", + "dlrVerificationPage": "http://localhost:3003/verify", + "dlrQualifierPath": "" + }, + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "transformationEventCredential": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/epcList/index/name", + "destinationPath": "/inputEPCList/index/name" + }, + { + "sourcePath": "/vc/credentialSubject/epcList/index/itemID", + "destinationPath": "/inputEPCList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/inputQuantityList", + "data": [ + { + "epcClass": "urn:epc:class:lgtin:4012345.012345.1000", + "quantity": 500, + "uom": "KGM" + }, + { + "epcClass": "urn:epc:class:lgtin:4012345.012345.1001", + "quantity": 300, + "uom": "KGM" + } + ] + }, + { + "path": "/outputQuantityList", + "data": [ + { + "epcClass": "urn:epc:class:lgtin:4012345.012345.2000", + "quantity": 200, + "uom": "KGM" + } + ] + }, + { + "path": "/outputEPCList", + "data": [{ "name": "0109359502000034" }] + }, + { + "path": "/action", + "data": "observe" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizLocation", + "data": "https://example.com/location/warehouse" + } + ], + "generationFields": [ + { + "path": "/outputEPCList/index/name", + "handler": "generateIdWithBatchLot" + }, + { + "path": "/outputEPCList/index/itemID", + "handler": "generateLinkResolver", + "dependencies": ["/outputEPCList/index/name"] + }, + { + "path": "/eventID", + "handler": "generateUUID" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + }, + "dppCredentials": [ + { + "mappingFields": [ + { + "sourcePath": "/linkResolver", + "destinationPath": "/traceabilityInformation/0/eventReference" + }, + { + "sourcePath": "/vc/credentialSubject/outputEPCList/0/name", + "destinationPath": "/productIdentifier/0/identifierValue" + }, + { + "sourcePath": "/vc/credentialSubject/outputEPCList/0/itemID", + "destinationPath": "/productIdentifier/0/binding/reference" + } + ], + "dummyFields": [ + { + "path": "/traceabilityInformation/0/eventType", + "data": "transformation" + }, + { + "path": "/batchIdentifier", + "data": [ + { + "scheme": "https://batch.example.org", + "identifierValue": "BATCH-77889", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://batch.example.org/77889/binding" + } + } + ] + }, + { + "path": "/itemIdentifier", + "data": [ + { + "scheme": "https://item.example.org", + "identifierValue": "ITEM-998877", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://item.example.org/998877/binding" + } + } + ] + }, + { + "path": "/classification", + "data": [ + { + "scheme": "https://unstats.un.org/unsd/classifications/Econ/cpc", + "classifierValue": "31254", + "classifierName": "Pile Cage", + "classifierURL": "https://vocabulary.uncefact.org/unlocode#USNYC" + } + ] + }, + { + "path": "/modelName", + "data": "Pile Cage" + }, + { + "path": "/image", + "data": "" + }, + { + "path": "/description", + "data": "High-strength pile cage suitable for construction and industrial applications." + }, + { + "path": "/furtherInformation", + "data": "https://example.com/products/PileCage" + }, + { + "path": "/manufacturedDate", + "data": "2024-05-15" + }, + { + "path": "/dimension", + "data": { + "weight": { + "value": 1500, + "unit": "KG" + }, + "length": { + "value": 2.44, + "unit": "MTR" + }, + "width": { + "value": 1.22, + "unit": "MTR" + }, + "height": { + "value": 0.018, + "unit": "MTR" + }, + "volume": { + "value": 0.053, + "unit": "CMT" + } + } + }, + { + "path": "/manufacturer", + "data": { + "id": "did:example:123456789abcdefghi", + "name": "SteelWorks Ltd.", + "identifiers": [ + { + "scheme": "https://example.com/company-scheme", + "identifierValue": "SW-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/company/001/binding" + } + } + ] + } + }, + { + "path": "/manufacturedAt", + "data": { + "identifier": [ + { + "scheme": "https://example.com/facility-scheme", + "identifierValue": "FAC-002", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/facility/002/binding" + } + } + ], + "name": "SteelWorks Manufacturing Plant 2", + "location": "https://plus.codes/4RRG5V3W+28", + "operatedBy": { + "id": "did:example:123456789abcdefghi", + "name": "SteelWorks Ltd.", + "identifiers": [ + { + "scheme": "https://example.com/company-scheme", + "identifierValue": "SW-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/company/001/binding" + } + } + ] + } + } + }, + { + "path": "/materialsProvenance", + "data": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://example.com/material-scheme", + "classifierValue": "steel", + "classifierName": "Steel" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ] + }, + { + "path": "/conformityClaim", + "data": [ + { + "topic": "Structural Steel Certification", + "standardOrRegulation": "https://steel-industry.org/standards/sustainability", + "criteriaReference": "https://steel-industry.org/standards/sustainability/criteria", + "claimedValues": [], + "benchmarkValue": {}, + "benchmarkReference": "https://steel-industry.org/benchmarks/sustainability", + "conformance": true, + "conformityEvidence": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://acrs.pyx.io/verify?q=%7B%22payload%22%3A%7B%22uri%22%3A%22https%3A%2F%2Fstorage.googleapis.com%2Fverifiable-credentials%2Fconformity-credentials%2Fsteel-processor-dcc.json%22%7D%7D" + } + } + ] + }, + { + "path": "/recyclingInstruction", + "data": "https://example.com/recycling/steelbeam-x200" + } + ] + } + ], + "identifierKeyPath": "/credentialSubject/outputEPCList/index/name" + } + ] + } + ] + } + ] + } + ], + "identifyProvider": { + "type": "gs1", + "url": "http://localhost:3001" + }, + "defaultVerificationServiceLink": { + "title": "Default Verification Service", + "context": "Default Verification Service", + "type": "application/json", + "href": "http://localhost:3332/agent/routeVerificationCredential", + "hreflang": ["en"], + "apiKey": "test123" + } +} diff --git a/app-config.json b/app-config.json index 31b0c812..27ce1358 100644 --- a/app-config.json +++ b/app-config.json @@ -1,5 +1,5 @@ { - "name": "Truffle Value Chain", + "name": "CHERRIES SUPPLY CHAIN TRACEABILITY", "styles": { "primaryColor": "rgb(35, 138, 186)", "secondaryColor": "black", @@ -19,11 +19,11 @@ ], "apps": [ { - "name": "Truffle Farm", + "name": "Orchard Facility", "type": "producer", "assets": { - "logo": "truffle-farm-logo.webp", - "brandTitle": "Truffle Farm" + "logo": "Cherries-farm-logo.webp", + "brandTitle": "Orchard Facility" }, "styles": { "primaryColor": "#b5651d", @@ -562,7 +562,7 @@ "productIdentifier": [ { "scheme": "https://id.gs1.org/gtin", - "identifierValue": "0105012345678900", + "identifierValue": "05012345678900", "binding": { "type": "document", "assuranceLevel": "3rdParty", @@ -572,23 +572,23 @@ ], "batchIdentifier": [ { - "scheme": "https://trufflefarm.example.org/batch", - "identifierValue": "BATCH-2024-001", + "scheme": "https://Cherriesfarm.example.org/batch", + "identifierValue": "2024001", "binding": { "type": "document", "assuranceLevel": "3rdParty", - "reference": "https://trufflefarm.example.org/batch/2024-001/binding" + "reference": "https://Cherriesfarm.example.org/batch/2024-001/binding" } } ], "itemIdentifier": [ { - "scheme": "https://trufflefarm.example.org/item", - "identifierValue": "TRF-24-0001", + "scheme": "https://Cherriesfarm.example.org/item", + "identifierValue": "TRF240001", "binding": { "type": "document", "assuranceLevel": "3rdParty", - "reference": "https://trufflefarm.example.org/item/TRF-24-0001/binding" + "reference": "https://Cherriesfarm.example.org/item/TRF-24-0001/binding" } } ], @@ -596,13 +596,13 @@ { "scheme": "https://www.unspsc.org", "classifierValue": "50383710", - "classifierName": "Truffles", + "classifierName": "Cherriess", "classifierURL": "https://www.unspsc.org/search-code/50383710" } ], - "modelName": "Black Truffle", - "description": "Premium Black Truffles harvested from our sustainable truffle orchards.", - "furtherInformation": "https://trufflefarm.example.org/products/black_perigord", + "modelName": "Black Cherries", + "description": "Premium Black Cherriess harvested from our sustainable Cherries orchards.", + "furtherInformation": "https://Cherriesfarm.example.org/products/black_perigord", "manufacturedDate": "2024-01-15", "dimension": { "weight": { "value": 50, "unit": "GRM" } @@ -613,7 +613,7 @@ "materialType": { "scheme": "https://www.gs1.org/gpc", "classifierValue": "10005953", - "classifierName": "Truffles (Fresh)", + "classifierName": "Cherriess (Fresh)", "classifierURL": "https://www.gs1.org/gpc/10005953" }, "massFraction": 100, @@ -629,7 +629,7 @@ }, "manufacturer": { "id": "did:example:123456789abcdefghi", - "name": "Gourmet Truffle Farm", + "name": "Gourmet Cherries Farm", "identifiers": [ { "scheme": "https://identifier.example.org/company", @@ -654,11 +654,11 @@ } } ], - "name": "Truffle Orchard", - "location": "https://trufflefarm.example.org/locations/perigord_orchard", + "name": "Cherries Orchard", + "location": "https://Cherriesfarm.example.org/locations/perigord_orchard", "operatedBy": { "id": "did:example:123456789abcdefghi", - "name": "Gourmet Truffle Farm", + "name": "Gourmet Cherries Farm", "identifiers": [ { "scheme": "https://identifier.example.org/company", @@ -673,6 +673,16 @@ } } }, + "constructData": { + "mappingFields": [], + "dummyFields": [], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + } + ] + }, "className": "json-form", "style": { "margin": "40px auto", @@ -694,7 +704,3722 @@ { "vckit": { "vckitAPIUrl": "http://localhost:3332/v1", - "issuer": "did:web:0e37-27-32-93-137.ngrok-free.app" + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Agriculture

Business identifier
75 859 224 235
Location
Rooty Hill, NSW
Identity verification
75 859 224 235
Other evidence

Conformity credentials

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Harvest Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Cherries Product Passport", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": { + "function": "concatService", + "args": [ + { "type": "text", "value": "(01)" }, + { "type": "path", "value": "/productIdentifier/0/identifierValue" }, + { "type": "text", "value": "(10)" }, + { "type": "path", "value": "/batchIdentifier/0/identifierValue" }, + { "type": "text", "value": "(21)" }, + { "type": "path", "value": "/itemIdentifier/0/identifierValue" } + ] + } + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "CherriesFarm_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Move to Next Facility", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "CherriesFarm_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Farm", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "CherriesFarm_dpps", "keyPath": "/epcList/index/name" } + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "orchard_facility_transaction_event", + "objectKeyPath": "/vc/credentialSubject/eventID" + } + ] + } + ] + } + ] + }, + { + "name": "Packhouse Facility", + "type": "producer", + "assets": { + "logo": "Cherries-farm-logo.webp", + "brandTitle": "Packhouse Facility" + }, + "styles": { + "primaryColor": "#b5651d", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "orchard_facility_transaction_event", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "image": "", + "productIdentifier": [ + { + "scheme": "https://id.gs1.org/gtin", + "identifierValue": "0105012345678900", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://id.gs1.org/gtin/05012345678900/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/batch", + "identifierValue": "BATCH-2024-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/batch/2024-001/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/item", + "identifierValue": "TRF-24-0001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/item/TRF-24-0001/binding" + } + } + ], + "classification": [ + { + "scheme": "https://www.unspsc.org", + "classifierValue": "50383710", + "classifierName": "Cherriess", + "classifierURL": "https://www.unspsc.org/search-code/50383710" + } + ], + "modelName": "Black Cherries", + "description": "Premium Black Cherriess harvested from our sustainable Cherries orchards.", + "furtherInformation": "https://Cherriesfarm.example.org/products/black_perigord", + "manufacturedDate": "2024-01-15", + "dimension": { + "weight": { "value": 50, "unit": "GRM" } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://www.gs1.org/gpc", + "classifierValue": "10005953", + "classifierName": "Cherriess (Fresh)", + "classifierURL": "https://www.gs1.org/gpc/10005953" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "characteristic": { + "variety": "Tuber melanosporum", + "grade": "Extra", + "aroma": "Intense, earthy", + "flavor": "Rich, complex" + }, + "manufacturer": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-5678", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/5678/binding" + } + } + ], + "name": "Cherries Orchard", + "location": "https://Cherriesfarm.example.org/locations/perigord_orchard", + "operatedBy": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + } + } + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/linkResolver", + "destinationPath": "/traceabilityInformation/0/eventReference" + } + ], + "dummyFields": [ + { + "path": "/traceabilityInformation/0/eventType", + "data": "transaction" + } + ], + "generationFields": [] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Agriculture

Business identifier
75 859 224 235
Location
Rooty Hill, NSW
Identity verification
75 859 224 235
Other evidence

Conformity credentials

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Harvest Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Cherries Product Passport", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/productIdentifier/0/identifierValue" + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "CherriesFarm_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Move to Next Facility", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "CherriesFarm_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Farm", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "CherriesFarm_dpps", "keyPath": "/epcList/index/name" } + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "packhouse_facility_transaction_event", + "objectKeyPath": "/vc/credentialSubject/eventID" + } + ] + } + ] + } + ] + }, + { + "name": "Fumigation and Freight Forwarding Facility", + "type": "producer", + "assets": { + "logo": "Cherries-farm-logo.webp", + "brandTitle": "Fumigation and Freight Forwarding Facility" + }, + "styles": { + "primaryColor": "#b5651d", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "packhouse_facility_transaction_event", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "image": "", + "productIdentifier": [ + { + "scheme": "https://id.gs1.org/gtin", + "identifierValue": "0105012345678900", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://id.gs1.org/gtin/05012345678900/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/batch", + "identifierValue": "BATCH-2024-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/batch/2024-001/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/item", + "identifierValue": "TRF-24-0001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/item/TRF-24-0001/binding" + } + } + ], + "classification": [ + { + "scheme": "https://www.unspsc.org", + "classifierValue": "50383710", + "classifierName": "Cherriess", + "classifierURL": "https://www.unspsc.org/search-code/50383710" + } + ], + "modelName": "Black Cherries", + "description": "Premium Black Cherriess harvested from our sustainable Cherries orchards.", + "furtherInformation": "https://Cherriesfarm.example.org/products/black_perigord", + "manufacturedDate": "2024-01-15", + "dimension": { + "weight": { "value": 50, "unit": "GRM" } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://www.gs1.org/gpc", + "classifierValue": "10005953", + "classifierName": "Cherriess (Fresh)", + "classifierURL": "https://www.gs1.org/gpc/10005953" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "characteristic": { + "variety": "Tuber melanosporum", + "grade": "Extra", + "aroma": "Intense, earthy", + "flavor": "Rich, complex" + }, + "manufacturer": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-5678", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/5678/binding" + } + } + ], + "name": "Cherries Orchard", + "location": "https://Cherriesfarm.example.org/locations/perigord_orchard", + "operatedBy": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + } + } + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/linkResolver", + "destinationPath": "/traceabilityInformation/0/eventReference" + } + ], + "dummyFields": [ + { + "path": "/traceabilityInformation/0/eventType", + "data": "transaction" + } + ], + "generationFields": [] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Agriculture

Business identifier
75 859 224 235
Location
Rooty Hill, NSW
Identity verification
75 859 224 235
Other evidence

Conformity credentials

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Harvest Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Cherries Product Passport", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/productIdentifier/0/identifierValue" + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "CherriesFarm_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Move to Next Facility", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "CherriesFarm_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Farm", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "CherriesFarm_dpps", "keyPath": "/epcList/index/name" } + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "fumigation_and_freight_forwarding_facility_transaction_event", + "objectKeyPath": "/vc/credentialSubject/eventID" + } + ] + } + ] + } + ] + }, + { + "name": "Airport Terminal Facility", + "type": "producer", + "assets": { + "logo": "Cherries-farm-logo.webp", + "brandTitle": "Airport Terminal" + }, + "styles": { + "primaryColor": "#b5651d", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "fumigation_and_freight_forwarding_facility_transaction_event", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "image": "", + "productIdentifier": [ + { + "scheme": "https://id.gs1.org/gtin", + "identifierValue": "0105012345678900", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://id.gs1.org/gtin/05012345678900/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/batch", + "identifierValue": "BATCH-2024-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/batch/2024-001/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://Cherriesfarm.example.org/item", + "identifierValue": "TRF-24-0001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://Cherriesfarm.example.org/item/TRF-24-0001/binding" + } + } + ], + "classification": [ + { + "scheme": "https://www.unspsc.org", + "classifierValue": "50383710", + "classifierName": "Cherriess", + "classifierURL": "https://www.unspsc.org/search-code/50383710" + } + ], + "modelName": "Black Cherries", + "description": "Premium Black Cherriess harvested from our sustainable Cherries orchards.", + "furtherInformation": "https://Cherriesfarm.example.org/products/black_perigord", + "manufacturedDate": "2024-01-15", + "dimension": { + "weight": { "value": 50, "unit": "GRM" } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://www.gs1.org/gpc", + "classifierValue": "10005953", + "classifierName": "Cherriess (Fresh)", + "classifierURL": "https://www.gs1.org/gpc/10005953" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "characteristic": { + "variety": "Tuber melanosporum", + "grade": "Extra", + "aroma": "Intense, earthy", + "flavor": "Rich, complex" + }, + "manufacturer": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-5678", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/5678/binding" + } + } + ], + "name": "Cherries Orchard", + "location": "https://Cherriesfarm.example.org/locations/perigord_orchard", + "operatedBy": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Cherries Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + } + } + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/linkResolver", + "destinationPath": "/traceabilityInformation/0/eventReference" + } + ], + "dummyFields": [ + { + "path": "/traceabilityInformation/0/eventType", + "data": "transaction" + } + ], + "generationFields": [] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" }, "dpp": { "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], @@ -705,13 +4430,15 @@ } ], "type": ["VerifiableCredential", "DigitalProductPassport"], - "dlrLinkTitle": "Truffle Product Passport", + "dlrLinkTitle": "Cherries Product Passport", "dlrIdentificationKeyType": "gtin", - "dlrVerificationPage": "http://localhost:3001/verify" + "dlrVerificationPage": "http://localhost:3003/verify" }, "dlr": { - "dlrAPIUrl": "http://localhost:8080", - "dlrAPIKey": "5555555555555" + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "storage": { "url": "http://localhost:3334/v1/documents", @@ -734,7 +4461,7 @@ "name": "mergeToLocalStorage", "parameters": [ { - "storageKey": "truffleFarm_dpps", + "storageKey": "CherriesFarm_dpps", "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" } ] @@ -742,14 +4469,14 @@ ] }, { - "name": "Sell Truffle", + "name": "Move to Next Facility", "id": "transaction_product", "components": [ { "name": "LocalStorageLoader", "type": "EntryData", "props": { - "storageKey": "truffleFarm_dpps", + "storageKey": "CherriesFarm_dpps", "nestedComponents": [ { "name": "JsonForm", @@ -1110,8 +4837,8 @@ { "path": "/sourceParty", "data": { - "id": "did:web:0e37-27-32-93-137.ngrok-free.app", - "name": "Truffle Farm", + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Farm", "identifiers": [ { "scheme": "https://example.com/scheme/source", @@ -1128,8 +4855,8 @@ { "path": "/destinationParty", "data": { - "id": "did:web:0e37-27-32-93-137.ngrok-free.app", - "name": "Truffle Processor", + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Cherries Processor", "identifiers": [ { "scheme": "https://example.com/scheme/destination", @@ -1175,7 +4902,7 @@ { "vckit": { "vckitAPIUrl": "http://localhost:3332/v1", - "issuer": "did:web:0e37-27-32-93-137.ngrok-free.app" + "issuer": "did:web:450e-103-69-79-17.ngrok-free.app" }, "epcisTransactionEvent": { "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], @@ -1188,11 +4915,13 @@ "type": ["TransactionEventCredential"], "dlrLinkTitle": "Transaction Event", "dlrIdentificationKeyType": "gtin", - "dlrVerificationPage": "http://localhost:3001/verify" + "dlrVerificationPage": "http://localhost:3003/verify" }, "dlr": { - "dlrAPIUrl": "http://localhost:8080", - "dlrAPIKey": "5555555555555" + "dlrAPIUrl": "http://localhost:3000", + "dlrAPIKey": "test123", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "storage": { "url": "http://localhost:3334/v1/documents", @@ -1208,7 +4937,7 @@ } }, "identifierKeyPath": "/eventID", - "localStorageParams": { "storageKey": "truffleFarm_dpps", "keyPath": "/epcList/index/name" } + "localStorageParams": { "storageKey": "CherriesFarm_dpps", "keyPath": "/epcList/index/name" } } ] } @@ -1219,7 +4948,8 @@ ], "identifyProvider": { "type": "gs1", - "url": "http://localhost:3333/products" + "url": "http://localhost:3001", + "namespace": "gs1" }, "defaultVerificationServiceLink": { "title": "Default Verification Service", diff --git a/app-config.truff.json b/app-config.truff.json new file mode 100644 index 00000000..bab932cc --- /dev/null +++ b/app-config.truff.json @@ -0,0 +1,1232 @@ +{ + "name": "Truffle Value Chain", + "styles": { + "primaryColor": "rgb(35, 138, 186)", + "secondaryColor": "black", + "tertiaryColor": "black" + }, + "generalFeatures": [ + { + "name": "General features", + "type": "", + "styles": { + "primaryColor": "rgb(35, 138, 186)", + "secondaryColor": "black", + "tertiaryColor": "black" + }, + "features": [] + } + ], + "apps": [ + { + "name": "Truffle Farm", + "type": "producer", + "assets": { + "logo": "truffle-farm-logo.webp", + "brandTitle": "Truffle Farm" + }, + "styles": { + "primaryColor": "#b5651d", + "secondaryColor": "#391561", + "tertiaryColor": "#ffffff" + }, + "features": [ + { + "name": "Issue DPP", + "id": "produce_product", + "components": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "Product": { + "type": "object", + "additionalProperties": false, + "properties": { + "productIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of unique identifiers assigned to the product or model. " + }, + "batchIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "Information regarding the specific production batch of the product." + }, + "itemIdentifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "An array of identifiers representing a specific serialised item of the product." + }, + "classification": { + "type": "array", + "items": { + "$ref": "#/$defs/Classification" + }, + "description": "A code representing the product's class, typically using the UN CPC (United Nations Central Product Classification) https://unstats.un.org/unsd/classifications/Econ/cpc" + }, + "modelName": { + "type": "string", + "description": "The model name or number of the product, represented as text." + }, + "image": { + "$ref": "#/$defs/BinaryFile", + "description": "A unique identifier (URI) pointing to an image of the product." + }, + "description": { + "type": "string", + "description": "A textual description providing details about the product." + }, + "furtherInformation": { + "type": "string", + "format": "uri", + "description": "A URL pointing to further human readable information about the product." + }, + "manufacturedDate": { + "type": "string", + "format": "date", + "description": "The ISO 8601 date on which the product batch was manufactured." + }, + "dimension": { + "$ref": "#/$defs/Dimension", + "description": "The physical dimensions of the product. Not every dimension is relevant to every products. For example bulk materials may have wieght and volume but not length, with, or height." + }, + "characteristic": { + "$ref": "#/$defs/Characteristic", + "description": "" + }, + "manufacturer": { + "$ref": "#/$defs/Party", + "description": "The Party entity that manufactured the product." + }, + "manufacturedAt": { + "$ref": "#/$defs/Facility", + "description": "The Facility where the product batch was manufactured." + }, + "materialsProvenance": { + "type": "array", + "items": { + "$ref": "#/$defs/Material" + }, + "description": "An array of Provenance objects providing details on the origin and mass fraction of components or ingredients of the product batch." + }, + "conformityClaim": { + "type": "array", + "items": { + "$ref": "#/$defs/Claim" + }, + "description": "An array of claim objects representing various product conformity claims about the product / batch. These can be sustainability claims, circularity claims, or any other claim type within the conformity topic list." + }, + "recyclingInstruction": { + "type": "string", + "format": "uri", + "description": "A URI pointing to information regarding the recycling aspects of the product." + }, + "traceabilityInformation": { + "type": "array", + "items": { + "$ref": "#/$defs/TraceabilityEvent" + }, + "description": "An array of TraceabilityEvent objects detailing EPCIS events related to the traceability of the product batch." + } + }, + "description": "The ProductInformation class encapsulates detailed information regarding a specific product, including its identification details, manufacturer, and other pertinent details." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Classification": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "Classification scheme - eg https://unstats.un.org/unsd/classifications/Econ/cpc " + }, + "classifierValue": { + "type": "string", + "description": "classifier value within the scheme - eg \"01211\" in UN CPC" + }, + "classifierName": { + "type": "string", + "description": "Name of the classifier - eg \"Asparagus\" for code \"01211\" in UNCPC" + }, + "classifierURL": { + "type": "string", + "format": "uri", + "description": "Linked data URL to a web vocabulary entery for this classificaiton code. When this property is provided, the scheme, value, and name properties of the classifer are not required. eg https://vocabulary.uncefact.org/unlocode#AUBNE represensign the port of Brisbane in the UN/LOCODE classification scheme." + } + }, + "description": "A classification scheme and code / name representing a category value for a product, entity, or facility." + }, + "BinaryFile": { + "type": "object", + "additionalProperties": false, + "properties": { + "fileHash": { + "x-jargon-isKey": true, + "type": "string", + "description": "The MD5 hash of the file." + }, + "fileLocation": { + "type": "string", + "format": "uri", + "description": "The location of the evidence file." + }, + "fileType": { + "type": "string", + "x-external-enumeration": "https://mimetype.io/all-types", + "description": "The type of file, represented as a MIME type.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://mimetype.io/all-types\n " + } + }, + "description": "A file representing a data snapshot that is used to infomr the conformity assessment." + }, + "Dimension": { + "type": "object", + "additionalProperties": false, + "properties": { + "weight": { + "$ref": "#/$defs/Measure", + "description": "the weight of the product" + }, + "length": { + "$ref": "#/$defs/Measure", + "description": "The length of the product or packaging" + }, + "width": { + "$ref": "#/$defs/Measure", + "description": "The width of the product or packaging" + }, + "height": { + "$ref": "#/$defs/Measure", + "description": "The height of the product or packaging" + }, + "volume": { + "$ref": "#/$defs/Measure", + "description": "The displacement volume of the product." + } + }, + "description": "Overall (length, width, height) dimensions and weight/volume of an item." + }, + "Measure": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "number", + "description": "The numeric value of the measure" + }, + "unit": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "Unit of measure drawn from the UNECE rec20 measure code list.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The measure class defines a numeric measured value (eg 10) and a coded unit of measure (eg KG)." + }, + "Characteristic": { + "type": "object", + "additionalProperties": false, + "properties": {}, + "description": "Product specific characteristics. This class is an extension point for industry specific product characteristics or performance information such as clothing size or battery capacity." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Facility": { + "type": "object", + "additionalProperties": false, + "properties": { + "identifier": { + "type": "array", + "items": { + "$ref": "#/$defs/Identifier" + }, + "description": "A unique identifier (URI) assigned to the facility. (Link Resolver - GS1 GLN?)" + }, + "name": { + "type": "string", + "description": "The name of the facility, represented as a text string." + }, + "location": { + "type": "string", + "format": "uri", + "description": "" + }, + "operatedBy": { + "$ref": "#/$defs/Party", + "description": "The Party entity responsible for operating the facility." + } + }, + "description": "The physical site (eg farm or factory) where the product or materials was produced." + }, + "Material": { + "type": "object", + "additionalProperties": false, + "properties": { + "originCountry": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/CountryId", + "description": "A ISO 3166-1 code representing the country of origin of the component or ingredient.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/CountryId\n " + }, + "materialType": { + "$ref": "#/$defs/Classification", + "description": "The type of this material - as a value drawn from a controlled vocabulary eg textileexchange.org/materials/rm01014 - representing organic cotton." + }, + "massFraction": { + "type": "number", + "description": "A numeric value representing the mass fraction of the product represented by this material. The sum of all mass fraction values for a given passport should be 100." + }, + "recycled": { + "type": "boolean", + "description": "Indicator is true if this material input is from a recycled source." + }, + "hazardous": { + "type": "boolean", + "description": "Indicates whether this material is hazardous. If true then " + } + }, + "description": "The material class encapsulates details about the origin or source of raw materials in a product, including the country of origin and the mass fraction." + }, + "Claim": { + "type": "object", + "additionalProperties": false, + "properties": { + "topic": { + "type": "string", + "enum": [ + "environment.energy", + "environment.emissions", + "environment.water", + "environment.waste", + "environment.deforestation", + "environment.biodiversity", + "circularity.content", + "circularity.design", + "social.labour", + "social.rights", + "social.community", + "social.safety", + "governance.ethics", + "governance.compliance", + "governance.transparency" + ], + "example": "environment.energy", + "description": "A code representing the topic of the sustainability claim. E.g. environment.deforestation, environment.ghg-emission-intensity, etc.. Drawn from a standard code list. " + }, + "standardOrRegulation": { + "type": "string", + "format": "uri", + "description": "The standard or regulation against which this conformity claim is made. Expressed as a URI and should match a value in the UN catalogue of reference vocabularies. " + }, + "criteriaReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the specific criteria within the standard or regulation against which this claim is made." + }, + "claimedValues": { + "type": "array", + "items": { + "$ref": "#/$defs/Metric" + }, + "description": "One or more actual measures supporting the claim. For example for GHG emissions there may be a metric for total emissions intensity and also a metric for amount of offsets included." + }, + "benchmarkValue": { + "$ref": "#/$defs/Metric", + "description": "A benchmark value against which the claimed value can be assessed. THis could be a value specified by a standard or regulation or could be an industry benchmark." + }, + "benchmarkReference": { + "type": "string", + "format": "uri", + "description": "A refernce to evidence to support the benchmark value." + }, + "conformance": { + "type": "boolean", + "description": "and indicator (boolean) that expresses whether or not this product has achieved compliance against the criteria. for example, if the topic is environment.deforstation and the criteria is EU.2023.1115 then the product is conformant if it has not touched any facility throughout it's lifecycle that is not deforestation free since dec 2020." + }, + "conformityEvidence": { + "$ref": "#/$defs/Evidence", + "description": "A URI pointing to the evidence supporting the claim. Most likely in the form of a verifiable credential." + } + }, + "description": "The SustainabilityClaim class represents specific claims regarding the sustainability of a product, providing details about the metrics, thresholds, and evidences supporting the claim." + }, + "Metric": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "description": "A human readable name for this metric" + }, + "value": { + "$ref": "#/$defs/Measure", + "description": "A numeric value representing the measurement or evaluation outcome for the claim." + }, + "accuracy": { + "type": "number", + "description": "A percentage represented as a numeric between 0 and 1 indicating the rage of accuracy of the claimed value (eg 0.05 means that the actual value is within 5% of the claimed value.)" + } + }, + "description": "A specific measure of performance against the criteria that governs the claim. Expressed as an array of metric (ie unit of emasure) / value (ie the actual numeric value) pairs. " + }, + "TraceabilityEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "eventReference": { + "type": "string", + "format": "uri", + "description": "A URI pointing to the detailed information about the EPCIS event. Most likely in the form of a verifiable credential." + }, + "eventType": { + "type": "string", + "enum": ["aggregation", "transformation", "object", "transaction", "association"], + "example": "aggregation", + "description": "A code representing the type of EPCIS event. ObjectEvent, AggregationEvent, TransactionEvent, TransformationEvent, ObjectEvent." + } + }, + "description": "The TraceabilityEvent class represents a specific EPCIS event in the traceability chain of a product, including details about the event type and reference." + } + } + }, + "data": { + "image": "", + "productIdentifier": [ + { + "scheme": "https://id.gs1.org/gtin", + "identifierValue": "0105012345678900", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://id.gs1.org/gtin/05012345678900/binding" + } + } + ], + "batchIdentifier": [ + { + "scheme": "https://trufflefarm.example.org/batch", + "identifierValue": "BATCH-2024-001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://trufflefarm.example.org/batch/2024-001/binding" + } + } + ], + "itemIdentifier": [ + { + "scheme": "https://trufflefarm.example.org/item", + "identifierValue": "TRF-24-0001", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://trufflefarm.example.org/item/TRF-24-0001/binding" + } + } + ], + "classification": [ + { + "scheme": "https://www.unspsc.org", + "classifierValue": "50383710", + "classifierName": "Truffles", + "classifierURL": "https://www.unspsc.org/search-code/50383710" + } + ], + "modelName": "Black Truffle", + "description": "Premium Black Truffles harvested from our sustainable truffle orchards.", + "furtherInformation": "https://trufflefarm.example.org/products/black_perigord", + "manufacturedDate": "2024-01-15", + "dimension": { + "weight": { "value": 50, "unit": "GRM" } + }, + "materialsProvenance": [ + { + "originCountry": "AU", + "materialType": { + "scheme": "https://www.gs1.org/gpc", + "classifierValue": "10005953", + "classifierName": "Truffles (Fresh)", + "classifierURL": "https://www.gs1.org/gpc/10005953" + }, + "massFraction": 100, + "recycled": false, + "hazardous": false + } + ], + "characteristic": { + "variety": "Tuber melanosporum", + "grade": "Extra", + "aroma": "Intense, earthy", + "flavor": "Rich, complex" + }, + "manufacturer": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Truffle Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + }, + "manufacturedAt": { + "identifier": [ + { + "scheme": "https://identifier.example.org/facility", + "identifierValue": "FAC-5678", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/facility/5678/binding" + } + } + ], + "name": "Truffle Orchard", + "location": "https://trufflefarm.example.org/locations/perigord_orchard", + "operatedBy": { + "id": "did:example:123456789abcdefghi", + "name": "Gourmet Truffle Farm", + "identifiers": [ + { + "scheme": "https://identifier.example.org/company", + "identifierValue": "COMP-12345", + "binding": { + "type": "document", + "assuranceLevel": "3rdParty", + "reference": "https://identifier.example.org/company/12345/binding" + } + } + ] + } + } + }, + "className": "json-form", + "style": { + "margin": "40px auto", + "paddingTop": "40px", + "width": "80%" + } + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": {} + } + ], + "services": [ + { + "name": "processDPP", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "dpp": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/dppld.json"], + "renderTemplate": [ + { + "template": " Digital Product Passport

PRODUCT PASSPORT

{{credentialSubject.modelName}}

{{#each credentialSubject.batchIdentifier}} {{/each}} {{#each credentialSubject.productIdentifier}} {{/each}}

Sustainability

100%

Confidence

100%

PASSPORT ISSUED BY

{{issuer.name}}

Industry

Agriculture

Business identifier
75 859 224 235
Location
Rooty Hill, NSW
Identity verification
75 859 224 235
Other evidence

Conformity credentials

Conformity credentials are usually issued by independent third parties and provide a trusted assessment of product ESG performance against credible standards or regulations

{{#each credentialSubject.conformityClaim}}

{{topic}}

{{!-- {{#if (eq conformityEvidence.type 'w3c_vc')}} --}}
Verifiable credential
{{!-- {{/if}} --}}

View details

{{/each}}

Product composition

A complete list of materials that make up the composition of this product.

{{#each credentialSubject.materialsProvenance}}

{{massFraction}}%

ID {{materialType.classifierValue}}

{{materialType.classifierName}}

{{#if recycled}}

Recycled

{{/if}} {{#if hazardous}}

Hazard

{{/if}}
{{originCountry}}
{{/each}}

Product information

Harvest Date

{{credentialSubject.manufacturedDate}}

Batch No.

{{credentialSubject.batchIdentifier.0.identifierValue}}

Weight

{{credentialSubject.dimension.weight.value}}{{credentialSubject.dimension.weight.unit}}

Description

{{credentialSubject.description}}

", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["VerifiableCredential", "DigitalProductPassport"], + "dlrLinkTitle": "Truffle Product Passport", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000/api", + "dlrAPIKey": "test123" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/productIdentifier/0/identifierValue" + } + ] + }, + { + "name": "mergeToLocalStorage", + "parameters": [ + { + "storageKey": "truffleFarm_dpps", + "objectKeyPath": "/vc/credentialSubject/productIdentifier/0/identifierValue" + } + ] + } + ] + }, + { + "name": "Sell Truffle", + "id": "transaction_product", + "components": [ + { + "name": "LocalStorageLoader", + "type": "EntryData", + "props": { + "storageKey": "truffleFarm_dpps", + "nestedComponents": [ + { + "name": "JsonForm", + "type": "EntryData", + "props": { + "schema": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller.", + "$schema": "https://json-schema.org/draft/2020-12/schema", + "$defs": { + "TransactionEvent": { + "type": "object", + "additionalProperties": false, + "properties": { + "sourceParty": { + "$ref": "#/$defs/Party", + "description": "The source party for this supply chain transaction - typically the seller party" + }, + "destinationParty": { + "$ref": "#/$defs/Party", + "description": "The destination party for this supply chain transaction - typically the buyer party." + }, + "epcList": { + "type": "array", + "items": { "$ref": "#/$defs/Item" }, + "description": "The list of uniquely identified trade items included in this supply chain transaction." + }, + "quantityList": { + "type": "array", + "items": { "$ref": "#/$defs/QuantityElement" }, + "description": "List of quantified product classes that are included in this transaction. Used when the trade items do not have unique identifiers (eg 100 reels of yarn)" + }, + "referenceDocument": { + "$ref": "#/$defs/TradeDocument", + "description": "The supply chain document reference for this transaction event - eg the invoice, order, or dispatch advice" + }, + "eventID": { + "x-jargon-isKey": true, + "readOnly": true, + "type": "string", + "description": "The unique identifier of this event - SHOULD be a UUID" + }, + "eventTime": { + "type": "string", + "format": "date-time", + "description": "The ISO-8601 date time when the event occurred." + }, + "action": { + "type": "string", + "enum": ["observe", "add", "delete"], + "example": "observe", + "description": "Code describing how an event relates to the lifecycle of the entity impacted by the event." + }, + "disposition": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/Disp", + "description": "Disposition code describing the state of the item after the event. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/Disp\n " + }, + "bizStep": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BizStep", + "description": "A business step code drawn from a controlled vocabulary. \n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BizStep\n " + }, + "bizLocation": { + "type": "string", + "format": "uri", + "description": "A Business Location is a uniquely identified and discretely recorded geospatial location that is meant to designate the specific place where an object is assumed to be following an EPCIS event until it is reported to be at a different Business Location by a subsequent EPCIS event. The bizLocation must be a resolvable URI that links to facility information and geolocation data." + }, + "sensorElementList": { + "type": "array", + "items": { "$ref": "#/$defs/SensorElement" }, + "description": "An array (one for each sensor) of sensor device data sets associated with the event. " + } + }, + "description": "Transaction represents an event in which one or more objects become associated or disassociated with one or more identified business transactions - such as the purchase / shipment of goods between buyer and seller." + }, + "Party": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "description": "The decentralised identifier of the party - must be a W3C DID." + }, + "name": { + "type": "string", + "description": "The name of the organization or company, represented as a text string." + }, + "identifiers": { + "type": "array", + "items": { "$ref": "#/$defs/Identifier" }, + "description": "A list of unique business identifiers assigned to the party - such as tax registration numbers." + } + }, + "description": "The Party class represents an entity such as an organization, or a company that manufactured the product." + }, + "Identifier": { + "type": "object", + "additionalProperties": false, + "properties": { + "scheme": { + "type": "string", + "format": "uri", + "description": "the identifier scheme as defined by the registrar that manages the identifier registry. If the identifier scheme is registered with UNTP then this URI can use used to dicsover the resolution method (to get more data) and the verification method (to prove ownership)." + }, + "identifierValue": { + "type": "string", + "description": "The value of the identifier within the scheme" + }, + "binding": { + "$ref": "#/$defs/Evidence", + "description": "Link to evidence that attests to the validity and ownership of the identifer. " + } + }, + "description": "An identifier of a party, product, or facility that is defined by an identifier scheme and idenfier value and, optinally, verification evidence " + }, + "Evidence": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "enum": ["w3c_vc", "iso_mdl", "document", "website", "other"], + "example": "w3c_vc", + "description": "Format of the evidence (verifiable credential, document, website, etc)" + }, + "assuranceLevel": { + "type": "string", + "enum": ["Self", "Commercial", "Buyer", "Membership", "Unspecified", "3rdParty"], + "example": "Self", + "description": "The assurance level of the evidence (self declaration, 2nd party, 3rd party, accredited auditor)" + }, + "reference": { + "type": "string", + "format": "uri", + "description": "The URL at which the evidence data can be found. " + } + }, + "description": "Evidence to support a conformity or identity claim. " + }, + "Item": { + "type": "object", + "additionalProperties": false, + "properties": { + "itemID": { + "x-jargon-isKey": true, + "type": "string", + "format": "uri", + "description": "The globally unique identifier (eg GS1 GTIN or digital link) of the product item. " + }, + "name": { + "type": "string", + "description": "The name of the product class to which the product item belongs. " + } + }, + "description": "A specific trade item /product code which could be either a product serial number or a consignment identifier " + }, + "QuantityElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "epcClass": { + "type": "string", + "format": "uri", + "description": "THe identifier of a product class (as opposed to a product instance) such as a GTIN code for a manufactured product." + }, + "quantity": { + "type": "number", + "description": "The numeric quantity of the product class (eg 100 kg of cotton)" + }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "The unit of measure for the quantity value (eg Kg or meters etc) using the UNECE Rec 20 unit of measure codelist.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "The quantity element is used to define the quantities (eg 100), units of measure (eg Kg) and product class (eg GTIN or other class identifier) of products that are inputs or outputs or the subject of supply chain events. ", + "required": ["quantity"] + }, + "TradeDocument": { + "type": "object", + "additionalProperties": false, + "properties": { + "type": { + "type": "string", + "x-external-enumeration": "https://ref.gs1.org/cbv/BTT", + "description": "The document type representing the trade transaction drawn from the business transaction type vocabulary.\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://ref.gs1.org/cbv/BTT\n " + }, + "identifier": { + "type": "string", + "description": "The identifier of the trade transaction document - eg an invoice number or bill of lading number. Must be unique for a given source party" + }, + "documentURL": { + "type": "string", + "format": "uri", + "description": "The URL of the referenced trade document. For integrity reasons, it is recommended (but not required) that the documentURL is a hashlink (https://w3c-ccg.github.io/hashlink/) so that if the document the URL is changed then the hash verification will fail." + } + }, + "description": "A trade transaction between two parties such as an invoice, purchase order, or shipping notification." + }, + "SensorElement": { + "type": "object", + "additionalProperties": false, + "properties": { + "sensorMetadata": { + "$ref": "#/$defs/Sensor", + "description": "Data that describes the physical sensor that recorded the sensor data set." + }, + "sensorReport": { + "type": "array", + "items": { "$ref": "#/$defs/SensorData" }, + "description": "A list of sensor readings from the given sensor relevant to the traceability event context." + }, + "sensorIntegrityProof": { + "type": "string", + "format": "uri", + "description": "An optional reference to a verifiable credential signed by the sensor device or device manufacturer that contains the digitally signed raw data associated with this sensor report." + } + }, + "description": "A SensorElement is used to carry data related to an event that is captured one sensor such as an IoT device. Include one sensor property and an array of sensor data values." + }, + "Sensor": { + "type": "object", + "additionalProperties": false, + "properties": { + "device": { + "$ref": "#/$defs/Item", + "description": "The device Identifier for the sensor as a URI (typically an EPC)" + }, + "dataProcessingMethod": { + "type": "string", + "format": "uri", + "description": "The data processing method used by the sensor - should reference a documented standard criteria as a URI" + } + }, + "description": "A physical sensor that records a sensor data set." + }, + "SensorData": { + "type": "object", + "additionalProperties": false, + "properties": { + "time": { + "type": "string", + "format": "date-time", + "description": "the timestamp at which the sensor reading was made." + }, + "type": { + "type": "string", + "format": "uri", + "description": "the measurement type of the sensor reading, as a URI reference to a measurement method specification." + }, + "value": { "type": "number", "description": "the sensor reading" }, + "uom": { + "type": "string", + "x-external-enumeration": "https://vocabulary.uncefact.org/UnitMeasureCode", + "description": "the unit of measure for the sensor reading\n\n This is an enumerated value, but the list of valid values are too big, or change too often to include here. You can access the list of allowable values at this URL: https://vocabulary.uncefact.org/UnitMeasureCode\n " + } + }, + "description": "A data point read by a sensor." + } + } + } + }, + "constructData": { + "mappingFields": [ + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/eventID" + }, + { + "sourcePath": "/vc/credentialSubject/productIdentifier/0/identifierValue", + "destinationPath": "/epcList/index/name" + }, + { + "sourcePath": "/linkResolver", + "destinationPath": "/epcList/index/itemID" + } + ], + "dummyFields": [ + { + "path": "/action", + "data": "observe" + }, + { + "path": "/disposition", + "data": "https://ref.gs1.org/cbv/Disp/in_transit" + }, + { + "path": "/bizStep", + "data": "https://ref.gs1.org/cbv/BizStep/receiving" + }, + { + "path": "/bizLocation", + "data": "https://example.com/warehouse" + }, + { + "path": "/sourceParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Truffle Farm", + "identifiers": [ + { + "scheme": "https://example.com/scheme/source", + "identifierValue": "SRC123456", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/source_evidence" + } + } + ] + } + }, + { + "path": "/destinationParty", + "data": { + "id": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app", + "name": "Truffle Processor", + "identifiers": [ + { + "scheme": "https://example.com/scheme/destination", + "identifierValue": "DST7891011", + "binding": { + "type": "w3c_vc", + "assuranceLevel": "3rdParty", + "reference": "https://example.com/destination_evidence" + } + } + ] + } + } + ], + "generationFields": [ + { + "path": "/eventID", + "handler": "generateIdWithSerialNumber" + }, + { + "path": "/eventTime", + "handler": "generateCurrentDatetime" + } + ] + } + } + ] + } + }, + { + "name": "CustomButton", + "type": "Submit", + "props": { + "includeDownload": true, + "downloadFileName": "transaction" + } + } + ], + "services": [ + { + "name": "processTransactionEvent", + "parameters": [ + { + "vckit": { + "vckitAPIUrl": "http://localhost:3332/v1", + "issuer": "did:web:143f-2406-2d40-4106-2b10-38c6-9732-f2d9-bb1c.ngrok-free.app" + }, + "epcisTransactionEvent": { + "context": ["https://dpp-json-ld.s3.ap-southeast-2.amazonaws.com/transaction-event-ld.json"], + "renderTemplate": [ + { + "template": "Transaction Event

TRACEABILITY EVENT

Transaction

EVENT ISSUED BY

{{issuer.name}}

{{#each issuer.identifiers}}
Industry
Needs to be replaced...
Business identifier
Needs to be replaced...
Identity verification
{{identiferValue}}
Verifiable credential
{{/each}}

Event description

Event ID
{{credentialSubject.eventID}}
Event type
Needs to be replaced...
Description
Needs to be replaced...
Time and date
{{credentialSubject.eventTime}}
Lifecycle action
{{credentialSubject.action}}
Product disposition
{{credentialSubject.disposition}}
Business step
{{credentialSubject.bizStep}}

Transaction

{{credentialSubject.sourceParty.name}}

SOURCE

{{credentialSubject.sourceParty.identifiers.0.identifierValue}}

Transferred

{{credentialSubject.destinationParty.name}}

DESTINATION

{{credentialSubject.destinationParty.partyID}}

Object list

{{#each credentialSubject.epcList}}

{{name}}

{{itemID}}

Product class name

Sustainability 0%
Confidence 0%
View
{{/each}}

Sensor

{{#each credentialSubject.sensorElementList}}
{{sensorMetadata.device.name}}
{{#each sensorReport}}

{{time}}

Data type

{{value}} {{uom}}

{{/each}}{{sensorIntegrityProof}}
Other evidence
{{/each}}
", + "@type": "WebRenderingTemplate2022" + } + ], + "type": ["TransactionEventCredential"], + "dlrLinkTitle": "Transaction Event", + "dlrIdentificationKeyType": "gtin", + "dlrVerificationPage": "http://localhost:3003/verify" + }, + "dlr": { + "dlrAPIUrl": "http://localhost:3000/api", + "dlrAPIKey": "test123" + }, + "storage": { + "url": "http://localhost:3334/v1/documents", + "params": { + "resultPath": "/uri", + "bucket": "verifiable-credentials" + }, + "options": { + "method": "POST", + "headers": { + "Content-Type": "application/json" + } + } + }, + "identifierKeyPath": "/eventID", + "localStorageParams": { "storageKey": "truffleFarm_dpps", "keyPath": "/epcList/index/name" } + } + ] + } + ] + } + ] + } + ], + "identifyProvider": { + "type": "gs1", + "url": "http://localhost:3001" + }, + "defaultVerificationServiceLink": { + "title": "Default Verification Service", + "context": "Default Verification Service", + "type": "application/json", + "href": "http://localhost:3332/agent/routeVerificationCredential", + "hreflang": ["en"], + "apiKey": "test123" + } +} diff --git a/docker-compose.yml b/docker-compose.yml index 9581fcff..7b1b93d4 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -47,5 +47,75 @@ services: - AVAILABLE_BUCKETS=verifiable-credentials,private-verifiable-credentials,epcis-events - STORAGE_TYPE=local + identity-resolver-service: + # Replace with local or deployed image + image: ghcr.io/pyx-industries/pyx-identity-resolver:latest@sha256:71a02b8b0a2b9c7cc18d33a54060e9af4d543e34bdd6a47c3b3cf4dcadb1711e + ports: + - '3000:3000' + environment: + - OBJECT_STORAGE_ENDPOINT=identity-resolver-service-object-store + - OBJECT_STORAGE_PORT=9000 + - OBJECT_STORAGE_USE_SSL=false + - OBJECT_STORAGE_ACCESS_KEY=minioadmin + - OBJECT_STORAGE_SECRET_KEY=minioadmin + - OBJECT_STORAGE_BUCKET_NAME=idr-bucket-1 + - OBJECT_STORAGE_PATH_STYLE=true + - IDENTIFIER_PATH=identifiers + - API_KEY=test123 + - APP_ENDPOINT=http://localhost:3000 + - APP_NAME=IDR-1 + - RESOLVER_DOMAIN=http://localhost:3000 + - LINK_TYPE_VOC_DOMAIN=http://localhost:3000/voc + - PORT=3000 + depends_on: + - identity-resolver-service-object-store + + identity-resolver-service-object-store: + image: quay.io/minio/minio:RELEASE.2024-08-17T01-24-54Z-cpuv1 + command: server /data --console-address ":9090" + ports: + - '9000:9000' + - '9090:9090' + environment: + - MINIO_ROOT_USER=minioadmin + - MINIO_ROOT_PASSWORD=minioadmin + volumes: + - ./minio_data/identity-resolver-service-object-store:/data + + mock-global-gs1-resolver: + # Replace with local or deployed image + image: ghcr.io/pyx-industries/pyx-identity-resolver:latest@sha256:71a02b8b0a2b9c7cc18d33a54060e9af4d543e34bdd6a47c3b3cf4dcadb1711e + ports: + - '3001:3001' + environment: + - OBJECT_STORAGE_ENDPOINT=mock-global-gs1-resolver-object-store + - OBJECT_STORAGE_PORT=9000 + - OBJECT_STORAGE_USE_SSL=false + - OBJECT_STORAGE_ACCESS_KEY=minioadmin + - OBJECT_STORAGE_SECRET_KEY=minioadmin + - OBJECT_STORAGE_BUCKET_NAME=idr-bucket-2 + - OBJECT_STORAGE_PATH_STYLE=true + - IDENTIFIER_PATH=identifiers + - API_KEY=test456 + - APP_ENDPOINT=http://localhost:3001 + - APP_NAME=IDR-2 + - RESOLVER_DOMAIN=http://localhost:3001 + - LINK_TYPE_VOC_DOMAIN=http://localhost:3001/voc + - PORT=3001 + depends_on: + - mock-global-gs1-resolver-object-store + + mock-global-gs1-resolver-object-store: + image: quay.io/minio/minio:RELEASE.2024-08-17T01-24-54Z-cpuv1 + command: server /data --console-address ":9091" + ports: + - '9001:9000' + - '9091:9090' + environment: + - MINIO_ROOT_USER=minioadmin + - MINIO_ROOT_PASSWORD=minioadmin + volumes: + - ./minio_data/mock-global-gs1-resolver-object-store:/data + volumes: - vckit-data: \ No newline at end of file + vckit-data: diff --git a/documentation/docs/mock-apps/common/identify-provider.md b/documentation/docs/mock-apps/common/identify-provider.md index bc2748e8..98802b3e 100644 --- a/documentation/docs/mock-apps/common/identify-provider.md +++ b/documentation/docs/mock-apps/common/identify-provider.md @@ -8,7 +8,8 @@ import Disclaimer from '../.././\_disclaimer.mdx'; ## Description -The `Identify Provider` object is a key component in the Mock App system that links scanned identifiers to their corresponding [Identity Resolver Service](/docs/mock-apps/dependent-services/identity-resolution-service), understands how to communicate with such services and encapsulates logic to interpret data retrieved from data carriers. + +The `Identify Provider` object is a key component in the Mock App system that links scanned identifiers to their corresponding [Identity Resolver Service](/docs/mock-apps/dependent-services/identity-resolution-service), understands how to communicate with such services and encapsulates logic to interpret data retrieved from data carriers. It serves three main functions: @@ -26,27 +27,29 @@ For instance, when dealing with a Global Trade Item Number (GTIN) from GS1: The Mock App system can use multiple identify providers, each tailored to a specific identity registrar and service. This modular approach allows the system to work with a variety of identification standards and services. ## Example + ```json { "identifyProvider": { "type": "gs1", - "url": "http://localhost:3333/products" + "url": "http://localhost:3333", + "namespace": "gs1" } } ``` ## Definitions -| Property | Required | Description | Type | -|----------|:--------:|-------------|------| -| type | Yes | The type of identify provider, e.g., "gs1" for GS1 standards. | [ProviderType](/docs/mock-apps/common/identify-provider#provider-types) | -| url | Yes | The URL endpoint for the identify provider service. | String | - +| Property | Required | Description | Type | +| --------- | :------: | ------------------------------------------------------------- | ----------------------------------------------------------------------- | +| type | Yes | The type of identify provider, e.g., "gs1" for GS1 standards. | [ProviderType](/docs/mock-apps/common/identify-provider#provider-types) | +| url | Yes | The URL endpoint for the identify provider service. | String | +| namespace | Yes | The namespace for the identify provider. | String | ## Provider Types -| Type | Description | -|------|-------------| -| gs1 | Used for resolving (using the Mock Verified By GS1 Service)/understanding GS1-based identifiers (e.g., GTINs). | +| Type | Description | +| ---- | -------------------------------------------------------------------------------------------------------------- | +| gs1 | Used for resolving (using the Mock Verified By GS1 Service)/understanding GS1-based identifiers (e.g., GTINs). | -**Note**: The available types may be extended in the future to support additional identity providers. \ No newline at end of file +**Note**: The available types may be extended in the future to support additional identity providers. diff --git a/documentation/docs/mock-apps/common/idr.md b/documentation/docs/mock-apps/common/idr.md index 5d5ac8d0..18763c94 100644 --- a/documentation/docs/mock-apps/common/idr.md +++ b/documentation/docs/mock-apps/common/idr.md @@ -13,7 +13,9 @@ The `IDR` object contains configuration for the [Identity Resolver Service](/doc ## Definition -| Property | Required | Description | Type | -|----------|----------|-------------|------| -| dlrAPIUrl | Yes | URL for the Identity Resolver API | String | -| dlrAPIKey | No | API key for the Identity Resolver | String | \ No newline at end of file +| Property | Required | Description | Type | +| ---------------- | -------- | ----------------------------------- | ------ | +| dlrAPIUrl | Yes | URL for the Identity Resolver API | String | +| dlrAPIKey | No | API key for the Identity Resolver | String | +| namespace | Yes | Namespace for the Identity Resolver | String | +| linkRegisterPath | No | Path to register a link | String | diff --git a/documentation/docs/mock-apps/services/process-aggregation-event.md b/documentation/docs/mock-apps/services/process-aggregation-event.md index abe58673..ae30e370 100644 --- a/documentation/docs/mock-apps/services/process-aggregation-event.md +++ b/documentation/docs/mock-apps/services/process-aggregation-event.md @@ -64,7 +64,9 @@ P-->>C: Return VC and resolver URL }, "dlr": { "dlrAPIUrl": "https://dlr.example.com/api", - "dlrAPIKey": "dlr-api-key-12345" + "dlrAPIKey": "dlr-api-key-12345", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "identifierKeyPath": "/parentItem/epc" } diff --git a/documentation/docs/mock-apps/services/process-dpp.md b/documentation/docs/mock-apps/services/process-dpp.md index c6595c74..923e600e 100644 --- a/documentation/docs/mock-apps/services/process-dpp.md +++ b/documentation/docs/mock-apps/services/process-dpp.md @@ -75,7 +75,9 @@ P-->>C: Return VC and resolver URL }, "dlr": { "dlrAPIUrl": "https://dlr.example.com", - "dlrAPIKey": "5555555555555" + "dlrAPIKey": "5555555555555", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "storage": { "url": "https://storage.example.com/v1/documents", diff --git a/documentation/docs/mock-apps/services/process-transaction-event.md b/documentation/docs/mock-apps/services/process-transaction-event.md index 999a1453..ea62c38a 100644 --- a/documentation/docs/mock-apps/services/process-transaction-event.md +++ b/documentation/docs/mock-apps/services/process-transaction-event.md @@ -66,7 +66,9 @@ P-->>C: Return VC and resolver URL }, "dlr": { "dlrAPIUrl": "https://dlr.example.com/api", - "dlrAPIKey": "dlr-api-key-12345" + "dlrAPIKey": "dlr-api-key-12345", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "identifierKeyPath": "/transactionId", "localStorageParams": { diff --git a/documentation/docs/mock-apps/services/process-transformation-event.md b/documentation/docs/mock-apps/services/process-transformation-event.md index 5b366022..0ec91972 100644 --- a/documentation/docs/mock-apps/services/process-transformation-event.md +++ b/documentation/docs/mock-apps/services/process-transformation-event.md @@ -65,7 +65,9 @@ P-->>C: Return EPCIS VC }, "dlr": { "dlrAPIUrl": "https://dlr.example.com/api", - "dlrAPIKey": "dlr-api-key-12345" + "dlrAPIKey": "dlr-api-key-12345", + "namespace": "gs1", + "linkRegisterPath": "/api/resolver" }, "storage": { "url": "https://storage.example.com/upload", diff --git a/packages/components/src/components/BarcodeGenerator/BarcodeGenerator.tsx b/packages/components/src/components/BarcodeGenerator/BarcodeGenerator.tsx index 410c2843..813b6455 100644 --- a/packages/components/src/components/BarcodeGenerator/BarcodeGenerator.tsx +++ b/packages/components/src/components/BarcodeGenerator/BarcodeGenerator.tsx @@ -1,5 +1,5 @@ -import { allowedIndexKeys } from '@mock-app/services'; -import { Box, Container } from '@mui/material'; +import { allowedIndexKeys, extractFromElementString } from '@mock-app/services'; +import { Box } from '@mui/material'; import JSONPointer from 'jsonpointer'; import { useEffect, useState } from 'react'; import Barcode from 'react-barcode'; @@ -15,7 +15,7 @@ export const BarcodeGenerator = (props: IBarcodeProps) => { const pathIndex = props.dataPath.split('/').findIndex((key) => allowedIndexKeys.includes(key)); if (pathIndex === -1) { - setValues(JSONPointer.get(props.data, props.dataPath)); + setValues([constructBarcode(JSONPointer.get(props.data, props.dataPath))]); } else { const headPath = props.dataPath.split('/').slice(0, pathIndex).join('/'); const tailPath = props.dataPath @@ -24,10 +24,46 @@ export const BarcodeGenerator = (props: IBarcodeProps) => { .join('/'); const array = JSONPointer.get(props.data, headPath); const values = array.map((item: any) => JSONPointer.get(item, `/${tailPath}`)); - setValues(values); + const parsedValues = values.map((item: any) => { + return constructBarcode(item); + }); + + setValues(parsedValues); } } }, [props.data, props.dataPath]); + + const constructBarcode = (data: string) => { + const convertToGS1String = (obj: any) => { + if (typeof obj !== 'object' || obj === null) return ''; + + let result = ''; + + // Handle '01' (01) first if it exists + if ('01' in obj) { + let value = obj['01']; + if (!/^\d{12,14}|\d{8}$/.test(value)) throw new Error('Invalid GTIN: ' + value); + value = value.padStart(14, '0'); + result += `(01)${value}`; + } + + // Handle all other AIs + for (const [ai, value] of Object.entries(obj) as any) { + if (ai === '01') continue; // Skip '01' as it's already handled + + if (!/^\d{2,4}$/.test(ai)) throw new Error(`Invalid AI: ${ai}`); + + result += `(${ai})${value}`; + } + + return result; + }; + + const aiArray = extractFromElementString(data); + + return convertToGS1String(aiArray); + }; + return ( {values.map((item) => { diff --git a/packages/mock-app/src/pages/Scanning.tsx b/packages/mock-app/src/pages/Scanning.tsx index b4d5615b..2b70855c 100644 --- a/packages/mock-app/src/pages/Scanning.tsx +++ b/packages/mock-app/src/pages/Scanning.tsx @@ -22,7 +22,7 @@ const Scanning = () => { try { setIsLoading(true); - const dlrUrl = await identityProvider.getDlrUrl(scannedCode); + const dlrUrl = await identityProvider.getDlrUrl(scannedCode, appConfig.identifyProvider.namespace); if (!dlrUrl) { return toastMessage({ status: Status.error, message: 'There no DLR url' }); } @@ -96,7 +96,7 @@ const Scanning = () => { qrbox={{ width: 500, height: 300 }} disableFlip={false} useBarCodeDetectorIfSupported={true} - focusMode= 'continuous' + focusMode='continuous' qrCodeSuccessCallback={onScanResult} qrCodeErrorCallback={onScanError} /> diff --git a/packages/services/src/__tests__/gs1.test.ts b/packages/services/src/__tests__/gs1.test.ts index 118a2aea..32faebe2 100644 --- a/packages/services/src/__tests__/gs1.test.ts +++ b/packages/services/src/__tests__/gs1.test.ts @@ -9,8 +9,9 @@ jest.mock('../types/types', () => ({ describe('Gs1Provider', () => { const gtinAI = 'gtin'; - const mockCode = '0109359502000010'; + const mockCode = '(01)09359502000010'; const providerUrl = 'https://example.com'; + const mockNamespace = 'gs1'; let gs1Provider: GS1Provider; @@ -20,79 +21,81 @@ describe('Gs1Provider', () => { }); describe('getDlrUrl', () => { - it('should return null if code is not set', async () => { - const code = ''; - jest.spyOn(publicAPI, 'post').mockResolvedValueOnce([]); - // Act - const dlrUrl = await gs1Provider.getDlrUrl(code, providerUrl); - - // Assert - expect(dlrUrl).toBeNull(); + it('should throw error if code is not set', async () => { + await expect(gs1Provider.getDlrUrl('', providerUrl, mockNamespace)).rejects.toThrow( + 'Failed to run get DLR Url. GTIN not found in the GS1 payload', + ); }); it('should return null if no products are fetched', async () => { - jest.spyOn(publicAPI, 'post').mockResolvedValueOnce([]); + jest.spyOn(publicAPI, 'get').mockResolvedValueOnce([]); - const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl); + const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl, mockNamespace); expect(dlrUrl).toBeNull(); }); it('should return null if gs1ServiceHost is not found', async () => { - const mockProducts = [{ - linkset: { - [GS1ServiceEnum.serviceInfo]: [] + const mockProducts = [ + { + linkset: { + [GS1ServiceEnum.serviceInfo]: [], + }, }, - }]; - jest.spyOn(publicAPI, 'post').mockResolvedValueOnce(mockProducts); + ]; + jest.spyOn(publicAPI, 'get').mockResolvedValueOnce(mockProducts); // Call the getDlrUrl method - const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl); + const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl, mockNamespace); // Ensure that the returned DLR URL is null expect(dlrUrl).toBeNull(); }); it('should return null if fetch fails', async () => { - jest.spyOn(publicAPI, 'post').mockRejectedValueOnce(new Error('Failed to fetch')); + jest.spyOn(publicAPI, 'get').mockRejectedValueOnce(new Error('Failed to fetch')); - await expect(gs1Provider.getDlrUrl(mockCode, providerUrl)).rejects.toThrow('Failed to run get DLR Url. Failed to fetch'); + await expect(gs1Provider.getDlrUrl(mockCode, providerUrl, mockNamespace)).rejects.toThrow( + 'Failed to run get DLR Url. Failed to fetch', + ); }); it('should return DLR URL if gs1ServiceHost is found', async () => { // Set a code, mock the post method to return products with gs1ServiceHost, and specify the mock GS1 host const mockGs1Host = 'https://gs1servicehost.com'; - const mockProducts = [{ - [gtinAI]: mockCode, - linkset: { - [GS1ServiceEnum.serviceInfo]: [{ href: mockGs1Host }] - }, - }]; - jest.spyOn(publicAPI, 'post').mockResolvedValueOnce(mockProducts); + const mockProducts = { + linkset: [ + { + [`${providerUrl}/${GS1ServiceEnum.serviceInfo}`]: [{ href: mockGs1Host }], + }, + ], + }; + jest.spyOn(publicAPI, 'get').mockResolvedValueOnce(mockProducts); // Call the getDlrUrl method - const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl); + const dlrUrl = await gs1Provider.getDlrUrl(mockCode, providerUrl, mockNamespace); // Ensure that the returned DLR URL matches the expected format - expect(dlrUrl).toBe(`${mockGs1Host}/gtin/${mockCode.slice(2)}?linkType=all`); + expect(dlrUrl).toBe(`${mockGs1Host}/gtin/${mockCode.slice(4)}?linkType=all`); }); it('should return DLR URL if the element string is combined multi AIs', async () => { const lotAI = '10'; const lotValue = '3000189'; const mockGs1Host = 'https://gs1servicehost.com'; - const elementStrings = `${mockCode}${lotAI}${lotValue}`; - const mockProducts = [{ - [gtinAI]: elementStrings, - linkset: { - [GS1ServiceEnum.serviceInfo]: [{ href: mockGs1Host }] - }, - }]; - jest.spyOn(publicAPI, 'post').mockResolvedValueOnce(mockProducts); - - const dlrUrl = await gs1Provider.getDlrUrl(elementStrings, providerUrl); - - expect(dlrUrl).toBe(`${mockGs1Host}/gtin/${mockCode.slice(2)}/lot/${lotValue}?linkType=all`); + const elementStrings = `${mockCode}(${lotAI})${lotValue}`; + const mockProducts = { + linkset: [ + { + [`${providerUrl}/${GS1ServiceEnum.serviceInfo}`]: [{ href: mockGs1Host }], + }, + ], + }; + jest.spyOn(publicAPI, 'get').mockResolvedValueOnce(mockProducts); + + const dlrUrl = await gs1Provider.getDlrUrl(elementStrings, providerUrl, mockNamespace); + + expect(dlrUrl).toBe(`${mockGs1Host}/gtin/${mockCode.slice(4)}/lot/${lotValue}?linkType=all`); }); }); @@ -121,5 +124,4 @@ describe('Gs1Provider', () => { expect(extractedCode).toBe('12345678901234'); }); }); - }); diff --git a/packages/services/src/__tests__/helpers.test.ts b/packages/services/src/__tests__/helpers.test.ts index 08e7bdc6..06ad667a 100644 --- a/packages/services/src/__tests__/helpers.test.ts +++ b/packages/services/src/__tests__/helpers.test.ts @@ -5,6 +5,7 @@ import { incrementQuality, concatService, constructIdentifierString, + extractDomain, } from '../utils/helpers'; describe('helpers', () => { @@ -44,6 +45,21 @@ describe('helpers', () => { }); }); +describe('extractDomain', () => { + it('should return the domain of a url', () => { + const url = 'https://example.com/test'; + const result = extractDomain(url); + expect(result).toBe('https://example.com'); + }); + + it('should return an empty string if the url is invalid', () => { + const url = 'invalid'; + expect(() => { + extractDomain(url); + }).toThrow('Invalid URL'); + }); +}); + describe('concatService', () => { it('should concatenate the values of the arguments passed to it', () => { const data = { diff --git a/packages/services/src/__tests__/linkResolver.test.ts b/packages/services/src/__tests__/linkResolver.test.ts index 59c1f588..9f24325e 100644 --- a/packages/services/src/__tests__/linkResolver.test.ts +++ b/packages/services/src/__tests__/linkResolver.test.ts @@ -37,6 +37,7 @@ describe('create link resolve service', () => { linkType: LinkType.epcisLinkType, dlrAPIUrl: 'https://dlr.com', dlrAPIKey: 'dlr-key', + namespace: 'gtin', qualifierPath: '', }; @@ -49,16 +50,12 @@ describe('create link resolve service', () => { mockValue.verificationPage, mockValue.dlrAPIUrl, mockValue.dlrAPIKey, + mockValue.namespace, ); expect(resolverUrl).toEqual( - `${mockValue.dlrAPIUrl}/${mockValue.identificationKeyType}/${mockValue.identificationKey}?linkType=all`, + `${mockValue.dlrAPIUrl}/${mockValue.namespace}/${mockValue.identificationKeyType}/${mockValue.identificationKey}?linkType=all`, ); - expect(mockValue.identificationKeyType).toEqual(expectParamsCallAPI[0].identificationKeyType); - expect(mockValue.identificationKey).toEqual(expectParamsCallAPI[0].identificationKey); - expect(mockValue.itemDescription).toEqual(expectParamsCallAPI[0].itemDescription); - expect(mockValue.verificationPage).toEqual(expectParamsCallAPI[0].responses[0].targetUrl); - expect(mockValue.dlrAPIKey).toEqual(expectToken); }); it('should throw error when creating link resolver', async () => { @@ -75,6 +72,7 @@ describe('create link resolve service', () => { qualifierPath: '', dlrAPIUrl: 'https://dlr.com', dlrAPIKey: 'dlr-key', + namespace: 'gtin', }); } catch (error: any) { expect(error.message).toEqual(errorMessage); diff --git a/packages/services/src/epcisEvents/aggregationEvent.ts b/packages/services/src/epcisEvents/aggregationEvent.ts index 74b7722e..0df1c8db 100644 --- a/packages/services/src/epcisEvents/aggregationEvent.ts +++ b/packages/services/src/epcisEvents/aggregationEvent.ts @@ -60,6 +60,7 @@ export const processAggregationEvent: IService = async ( epcisAggregationEvent.dlrVerificationPage, dlr.dlrAPIUrl, dlr.dlrAPIKey, + dlr.namespace, qualifierPath, ); diff --git a/packages/services/src/epcisEvents/helpers.ts b/packages/services/src/epcisEvents/helpers.ts index 3f74e670..85f205e9 100644 --- a/packages/services/src/epcisEvents/helpers.ts +++ b/packages/services/src/epcisEvents/helpers.ts @@ -3,11 +3,16 @@ import { deleteValuesFromLocalStorage } from '../features/localStorage.service.j import { buildElementString, extractFromElementString, getLinkResolverIdentifier } from '../linkResolver.service.js'; import { ICurrentAndDependencies, allowedIndexKeys, randomIntegerString } from '../utils/helpers.js'; -export const generateLinkResolver = (dlrUrl: string, identificationKeyType: string, queryString: string) => { +export const generateLinkResolver = ( + dlrUrl: string, + namespace: string, + identificationKeyType: string, + queryString: string, +) => { const _generateLinkResolver = ({ currentData, dependenciesValues }: ICurrentAndDependencies) => { const { identifier, qualifierPath } = getLinkResolverIdentifier(dependenciesValues![0]); const path = qualifierPath.includes('?') ? `${qualifierPath}&${queryString}` : `${qualifierPath}?${queryString}`; - return `${dlrUrl}/${identificationKeyType}/${identifier}${path}`; + return `${dlrUrl}/${namespace}/${identificationKeyType}/${identifier}${path}`; }; return _generateLinkResolver; }; diff --git a/packages/services/src/epcisEvents/transactionEvent.ts b/packages/services/src/epcisEvents/transactionEvent.ts index 69c84b8b..1dd4246f 100644 --- a/packages/services/src/epcisEvents/transactionEvent.ts +++ b/packages/services/src/epcisEvents/transactionEvent.ts @@ -48,6 +48,7 @@ export const processTransactionEvent: IService = async ( epcisTransactionEvent.dlrVerificationPage, dlr.dlrAPIUrl, dlr.dlrAPIKey, + dlr.namespace, qualifierPath, LinkType.epcisLinkType, ); diff --git a/packages/services/src/epcisEvents/transformationEvent.ts b/packages/services/src/epcisEvents/transformationEvent.ts index 5c02858d..e24d4c99 100644 --- a/packages/services/src/epcisEvents/transformationEvent.ts +++ b/packages/services/src/epcisEvents/transformationEvent.ts @@ -83,6 +83,7 @@ export const processTransformationEvent: IService = async (data: any, context: I epcisTransformationEventContext.dlrVerificationPage, dlrContext.dlrAPIUrl, dlrContext.dlrAPIKey, + dlrContext.namespace, transformationEventQualifierPath, LinkType.epcisLinkType, ); @@ -105,6 +106,7 @@ export const processTransformationEvent: IService = async (data: any, context: I dppContext.dlrVerificationPage, dlrContext.dlrAPIUrl, dlrContext.dlrAPIKey, + dlrContext.namespace, qualifierPath, ); }), @@ -140,6 +142,7 @@ export const issueEpcisTransformationEvent = async ( handlers: { generateLinkResolver: generateLinkResolver( dlrContext.dlrAPIUrl, + dlrContext.namespace, IdentificationKeyType.gtin, `linkType=${LinkType.certificationLinkType}`, ), diff --git a/packages/services/src/epcisEvents/types.ts b/packages/services/src/epcisEvents/types.ts index b3cc46e8..19d8156d 100644 --- a/packages/services/src/epcisEvents/types.ts +++ b/packages/services/src/epcisEvents/types.ts @@ -32,6 +32,7 @@ export interface IDLRAI { export interface IConfigDLR { dlrAPIUrl: string; dlrAPIKey: string; + namespace: string; } export interface IStorageContext { diff --git a/packages/services/src/identityProviders/GS1Provider.ts b/packages/services/src/identityProviders/GS1Provider.ts index cff6cb38..3c35637d 100644 --- a/packages/services/src/identityProviders/GS1Provider.ts +++ b/packages/services/src/identityProviders/GS1Provider.ts @@ -1,11 +1,11 @@ import GS1DigitalLinkToolkit from 'GS1_DigitalLink_Resolver_CE/digitallink_toolkit_server/src/GS1DigitalLinkToolkit.js'; -import { IdentityProviderStrategy } from './IdentityProvider.js'; import { publicAPI } from '../utils/httpService.js'; +import { IdentityProviderStrategy } from './IdentityProvider.js'; export enum GS1ServiceEnum { - certificationInfo = 'https://gs1.org/voc/certificationInfo', - verificationService = 'https://gs1.org/voc/verificationService', - serviceInfo = 'https://gs1.org/voc/serviceInfo', + certificationInfo = 'voc/certificationInfo', + verificationService = 'voc/verificationService', + serviceInfo = 'voc/serviceInfo', } export class GS1Provider implements IdentityProviderStrategy { @@ -13,20 +13,45 @@ export class GS1Provider implements IdentityProviderStrategy { * Function to retrieve the DLR URL based on the GTIN code and identification provider URL. * @returns The DLR (Digital Link Resolver) URL corresponding to the provided GTIN code, or null if not found. */ - async getDlrUrl(code: string, providerUrl: string): Promise { + async getDlrUrl(code: string, providerUrl: string, namespace?: string): Promise { + const parseGS1Payload = (payload: any) => { + const aiRegex = /\((\d+)\)([^(]+)/g; + const parsed = Array.from(payload.matchAll(aiRegex), (match) => [(match as any)[1], (match as any)[2]]); + return parsed.flat().join('/'); + }; + try { - const fetchProductPayload = { keys: [code] }; - const products: any[] = await publicAPI.post(providerUrl, fetchProductPayload); - if (!products || !products.length) { + const fetchProductPayload = parseGS1Payload(code); + + const extractGTIN = (gs1String: string): string | null => { + const parts = gs1String.split('/'); + for (let i = 0; i < parts.length; i += 2) { + if (parts[i] === '01' && i + 1 < parts.length) { + return `01/${parts[i + 1]}`; + } + } + return null; + }; + + const gtin = extractGTIN(fetchProductPayload); + if (gtin === null) { + throw new Error('GTIN not found in the GS1 payload'); + } + + const { linkset }: any = await publicAPI.get( + namespace ? `${providerUrl}/${namespace}/${gtin}?linkType=all` : `${providerUrl}/${gtin}?linkType=all`, + ); + + if (!linkset || !linkset.length) { return null; } // Extract the GS1 service host from the fetched products data - const gs1ServiceHost: string = products[0]?.linkset?.[GS1ServiceEnum.serviceInfo]?.[0]?.href; + const gs1ServiceHost: string = linkset[0]?.[`${providerUrl}/${GS1ServiceEnum.serviceInfo}`]?.[0]?.href; if (!gs1ServiceHost) { return null; } - + const gs1DigitalLinkToolkit = new GS1DigitalLinkToolkit(); const gs1DigitalLink = gs1DigitalLinkToolkit.gs1ElementStringsToGS1DigitalLink(code, true, gs1ServiceHost); diff --git a/packages/services/src/identityProviders/IdentityProvider.ts b/packages/services/src/identityProviders/IdentityProvider.ts index 91088247..da6d91aa 100644 --- a/packages/services/src/identityProviders/IdentityProvider.ts +++ b/packages/services/src/identityProviders/IdentityProvider.ts @@ -1,5 +1,5 @@ export interface IdentityProviderStrategy { - getDlrUrl: (code: string, providerUrl: string) => Promise; + getDlrUrl: (code: string, providerUrl: string, namespace?: string) => Promise; getCode: (decodedText: string, formatName: string) => string; } @@ -12,8 +12,8 @@ export class IdentityProvider { this.identityProviderUrl = identityProviderUrl; } - getDlrUrl(code: string): Promise { - return this.identityProviderStrategy.getDlrUrl(code, this.identityProviderUrl); + getDlrUrl(code: string, namespace?: string): Promise { + return this.identityProviderStrategy.getDlrUrl(code, this.identityProviderUrl, namespace); } getCode(decodedText: string, formatName: string): string { diff --git a/packages/services/src/linkResolver.service.ts b/packages/services/src/linkResolver.service.ts index b0d5fa89..3c6300c7 100644 --- a/packages/services/src/linkResolver.service.ts +++ b/packages/services/src/linkResolver.service.ts @@ -3,6 +3,7 @@ import { IDLRAI } from './epcisEvents/types.js'; import { GS1ServiceEnum } from './identityProviders/GS1Provider.js'; import { MimeTypeEnum } from './types/types.js'; import { privateAPI } from './utils/httpService.js'; +import { extractDomain } from './utils/helpers.js'; /** * Generates a link resolver URL based on the provided linkResolver and linkResponse objects. * @@ -34,9 +35,9 @@ import { privateAPI } from './utils/httpService.js'; */ export enum LinkType { - verificationLinkType = 'gs1:verificationService', - certificationLinkType = 'gs1:certificationInfo', - epcisLinkType = 'gs1:epcis', + verificationLinkType = 'verificationService', + certificationLinkType = 'certificationInfo', + epcisLinkType = 'epcis', } export enum MimeType { @@ -74,18 +75,22 @@ export interface ICreateLinkResolver { qualifierPath: string; dlrAPIUrl: string; dlrAPIKey: string; + namespace: string; + linkRegisterPath?: string; responseLinkType?: string; queryString?: string | null; } -export interface GS1LinkResolver extends ILinkResolver { +export interface LinkResolver extends Omit { + namespace: string; qualifierPath: string; active: boolean; - responses: GS1LinkResponse[]; + responses: LinkResponse[]; } -export interface GS1LinkResponse extends ILinkResponse { +export interface LinkResponse extends ILinkResponse { + title: string; ianaLanguage: string; context: string; active: boolean; @@ -96,17 +101,20 @@ export interface GS1LinkResponse extends ILinkResponse { } export const createLinkResolver = async (arg: ICreateLinkResolver): Promise => { - const { dlrAPIUrl, linkResolver, linkResponses, qualifierPath, responseLinkType = 'all' } = arg; - let registerQualifierPath = qualifierPath; - if (arg.queryString) { - registerQualifierPath = qualifierPath.includes('?') - ? `${qualifierPath}&${arg.queryString}` - : `${qualifierPath}?${arg.queryString}`; - } - const params: GS1LinkResolver[] = [constructLinkResolver(linkResolver, linkResponses, registerQualifierPath)]; + const { + dlrAPIUrl, + namespace, + linkRegisterPath = '/api/resolver', + linkResolver, + linkResponses, + qualifierPath, + responseLinkType = 'all', + } = arg; + + const params: LinkResolver = constructLinkResolver(namespace, linkResolver, linkResponses, qualifierPath); try { privateAPI.setBearerTokenAuthorizationHeaders(arg.dlrAPIKey || ''); - await privateAPI.post(`${dlrAPIUrl}/resolver`, params); + await privateAPI.post(`${dlrAPIUrl}${linkRegisterPath}`, params); const path = responseLinkType === 'all' @@ -114,28 +122,30 @@ export const createLinkResolver = async (arg: ICreateLinkResolver): Promise { - const gs1LinkResolver: GS1LinkResolver = { + const constructedLinkResolver: LinkResolver = { + namespace: namespace, identificationKeyType: linkResolver.identificationKeyType, identificationKey: linkResolver.identificationKey, itemDescription: linkResolver.itemDescription, - qualifierPath, + qualifierPath: qualifierPath ? qualifierPath : '/', active: true, responses: [], }; linkResponses.forEach((linkResponse: ILinkResponse) => { - const gs1LinkResponseForUS: GS1LinkResponse = { + const LinkResponseForUS: LinkResponse = { ianaLanguage: 'en', context: 'us', defaultLinkType: false, @@ -144,10 +154,11 @@ export const constructLinkResolver = ( defaultMimeType: false, fwqs: false, active: true, + title: linkResponse.linkTitle, ...linkResponse, }; - const gs1LinkResponseForAU: GS1LinkResponse = { + const LinkResponseForAU: LinkResponse = { ianaLanguage: 'en', context: 'au', defaultLinkType: false, @@ -156,12 +167,13 @@ export const constructLinkResolver = ( defaultMimeType: false, fwqs: false, active: true, + title: linkResponse.linkTitle, ...linkResponse, }; - gs1LinkResolver.responses.push(gs1LinkResponseForUS, gs1LinkResponseForAU); + constructedLinkResolver.responses.push(LinkResponseForUS, LinkResponseForAU); }); - return gs1LinkResolver; + return constructedLinkResolver; }; export const registerLinkResolver = async ( @@ -173,6 +185,7 @@ export const registerLinkResolver = async ( verificationPage: string, dlrAPIUrl: string, dlrAPIKey: string, + namespace: string, qualifierPath?: string, responseLinkType?: string, ) => { @@ -186,19 +199,19 @@ export const registerLinkResolver = async ( const verificationPassportPage = `${verificationPage}/?${queryString}`; const linkResponses: ILinkResponse[] = [ { - linkType: LinkType.verificationLinkType, + linkType: `${namespace}:${LinkType.verificationLinkType}`, linkTitle: 'VCKit verify service', targetUrl: verificationPage, mimeType: MimeType.textPlain, }, { - linkType: linkType, + linkType: `${namespace}:${linkType}`, linkTitle: linkTitle, targetUrl: url, mimeType: MimeType.applicationJson, }, { - linkType: linkType, + linkType: `${namespace}:${linkType}`, linkTitle: linkTitle, targetUrl: verificationPassportPage, mimeType: MimeType.textHtml, @@ -210,6 +223,7 @@ export const registerLinkResolver = async ( return await createLinkResolver({ dlrAPIUrl, + namespace, linkResolver, linkResponses, queryString, @@ -224,6 +238,8 @@ export const registerLinkResolver = async ( * @returns The DLR passport data if found, otherwise returns null. */ export const getDlrPassport = async (dlrUrl: string): Promise => { + const rootDlrDomain = extractDomain(dlrUrl); + // Fetch DLR data from the provided DLR URL const dlrData = await privateAPI.get(dlrUrl); if (!dlrData) { @@ -232,14 +248,14 @@ export const getDlrPassport = async (dlrUrl: string): Promise => { // Find certificate passports in the DLR data const certificatePassports = dlrData?.linkset?.find( - (linkSetItem: any) => linkSetItem[GS1ServiceEnum.certificationInfo], + (linkSetItem: any) => linkSetItem[`${rootDlrDomain}/${GS1ServiceEnum.certificationInfo}`], ); if (!certificatePassports) { return null; } // Extract passport infos from certificate passports - const dlrPassports = certificatePassports[GS1ServiceEnum.certificationInfo]; + const dlrPassports = certificatePassports[`${rootDlrDomain}/${GS1ServiceEnum.certificationInfo}`]; if (!dlrPassports) { return null; } diff --git a/packages/services/src/processDPP.service.ts b/packages/services/src/processDPP.service.ts index 117befc3..dad3ad59 100644 --- a/packages/services/src/processDPP.service.ts +++ b/packages/services/src/processDPP.service.ts @@ -52,6 +52,7 @@ export const processDPP: IService = async (data: any, context: IContext): Promis dppContext.dlrVerificationPage, linkResolverContext.dlrAPIUrl, linkResolverContext.dlrAPIKey, + linkResolverContext.namespace, qualifierPath, LinkType.certificationLinkType, ); diff --git a/packages/services/src/utils/helpers.ts b/packages/services/src/utils/helpers.ts index b7bef623..9da3f849 100644 --- a/packages/services/src/utils/helpers.ts +++ b/packages/services/src/utils/helpers.ts @@ -301,6 +301,11 @@ export const constructObject = ( return data; }; +export const extractDomain = (url: string): string => { + const parsedUrl = new URL(url); + return `${parsedUrl.protocol}//${parsedUrl.host}`; +}; + export const constructIdentifierString = ( data: any, identifierKeyPath: string | { function: string; args: any },