Skip to main content

Check out Port for yourselfย 

Ingest dependencies from a package.json file and relate them to a service

Overviewโ€‹

This guide will demonstrate how to ingest dependencies from a package.json file and relate them to the corresponding service entities in Port.

Prerequisitesโ€‹

Set up data modelโ€‹

Add a dependency blueprintโ€‹

  1. Go to the Builder in your Port portal.

  2. Click on "+ Blueprint".

  3. Click on the {...} button in the top right corner, and choose "Edit JSON"

  4. Add this JSON schema:

    Dependency blueprint (Click to expand)
    {
    "identifier": "dependency",
    "title": "Dependency",
    "icon": "Package",
    "schema": {
    "properties": {
    "package_name": {
    "icon": "DefaultProperty",
    "type": "string",
    "title": "Package name"
    },
    "semver_requirement": {
    "type": "string",
    "title": "Semver requirement"
    },
    "type": {
    "type": "string",
    "title": "Type",
    "enum": [
    "Production",
    "Development"
    ]
    },
    "url": {
    "type": "string",
    "title": "URL",
    "format": "url"
    }
    },
    "required": [
    "package_name",
    "semver_requirement"
    ]
    },
    "mirrorProperties": {},
    "calculationProperties": {},
    "aggregationProperties": {},
    "relations": {}
    }

Ingest dependencies from package.jsonโ€‹

To ingest dependencies listed in package.json files, follow these steps:

  1. Go to the data sources page in your Port portal, and select your GitHub integration.

  2. Modify the mapping to include the file kind with the configuration provided below:

    Port Configuration (Click to expand)

    - kind: file
    selector:
    query: 'true'
    files:
    - path: '**/package.json'
    port:
    itemsToParse: .file.content.dependencies | to_entries
    entity:
    mappings:
    identifier: >-
    .item.key + "_" + (.item.value | gsub("\\^"; "caret_") |
    gsub("~"; "tilde_") | gsub(">="; "gte_") | gsub("<="; "lte_") |
    gsub(">"; "gt_") | gsub("<"; "lt_") | gsub("@"; "at_") |
    gsub("\\*"; "star") | gsub(" "; "_"))
    title: .item.key + "@" + .item.value
    blueprint: '"dependency"'
    properties:
    package_name: .item.key
    semver_requirement: .item.value
    Configuration Details
    • kind: file specifies that the source is a file, in this case, package.json.
    • files: defines the path pattern to locate package.json files within your repositories.
    • itemsToParse: identifies the specific array within the package.json (i.e., dependencies) that you want to parse into individual dependency entities.
    • identifier: constructs a unique identifier for each dependency, accounting for special characters in the version string.
    • properties: captures essential details like the package name and version.

Relate the dependencies to the serviceโ€‹

Once the dependencies have been ingested, the next step is to establish relationships between these dependency entities and the corresponding service entities.

  1. Go to the Builder in your Port portal, select the Service blueprint, and click on New relation to create a relation between the service and dependency blueprints.

  2. Click on the ... button in the top right corner of the Service blueprint and select Edit JSON.

  3. Add this JSON to establish the relationship:

      "dependencies": {
    "title": "Dependencies",
    "target": "dependency",
    "required": false,
    "many": true
    }
  4. Head back to the data sources page and add one of the following mapping approaches:

    The most straightforward way to set a relation's value is to explicitly specify the related entity's identifier:

      - kind: file
    selector:
    query: 'true'
    files:
    - path: '**/package.json'
    port:
    entity:
    mappings:
    identifier: .repo.name
    blueprint: '"service"'
    properties: {}
    relations:
    dependencies: >-
    [.file.content.dependencies | to_entries | map( .key + "_" +
    (.value |
    gsub("\\^"; "caret_") |
    gsub("~"; "tilde_") |
    gsub(">="; "gte_") |
    gsub("<="; "lte_") |
    gsub(">"; "gt_") |
    gsub("<"; "lt_") |
    gsub("@"; "at_") |
    gsub("\\*"; "star") |
    gsub(" "; "_")
    ) ) | .[]]
    Mapping Details

    This would establish a relation between the service and dependency entities based on the dependencies listed in the package.json file.

  5. After you add the mapping, click on the resync button and watch your repositories being mapped to their dependencies as shown below in this example:

Conclusionโ€‹

By following these steps, you can effectively ingest dependencies from package.json files and relate them to the corresponding repository entities in Port ๐ŸŽ‰.

More relevant guides and examples: