Create a value stream with Jira, Github, and Jenkins integrations and exercise it from beginning (a new issue in Backlog) to end (deployment to Prod).

We will start with the planning phase of our pipeline by focussing on Jira. We will need an instance of Jira to setup a Jira board and create a Jira card. We will create a Jira integration within Velocity and create a new value stream to work with this integration. We will finish the Jira section by seeing how changes to Jira cards are captured by an UrbanCode Velocity value stream.
As the issue transitions from the planning phase to the development phase we will focus on GitHub. We will need an instance of GitHub with a repository and pull request (PR). We will create a GitHub integration within Velocity and add this integration to a value stream. We will then observe work item progress all the way to “Merged” based on GitHub activity.
Lastly, we will turn our attention to the final phase of deployment. We will need a Jenkins instance for this equipped with an UrbanCode Velocity plugin. We will create a Jenkins integration within Velocity and define an application within our value streams pipeline. We will use jenkins jobs to build and deploy this application and observe the value stream as the work items continue on through build and deployments all the way to our production environment.
Questions or suggestions? Create an issue at https://github.com/urbancode/velocity-info/issues
Table of Contents
Our first order of business is to create the value stream. All we want to do is create the value stream right now. We will come back to configure it after setting up Jira.


| Field | Description | Required | 
|---|---|---|
| Name | value stream name. | yes | 
| Description | value stream description | no | 
| Team | Team of users who can access this value stream. | yes | 
Jira instance
This workbook requires a Jira instance that can be accessed and authenticated against using an API Token.
If you do not have a Jira instance readily available for this purpose, one option is to use a cloud Jira instance from https://www.atlassian.com.


Jira Board
For this workbook, your Jira board should have the five columns and status names listed below. The default Kanban board is a good place to start. You do not have to use Kanban though.

Jira Issue This workbook begins with a single issue in the “backlog” column.

See also Jira plugin Documentation at https://www.urbancode.com
Navigate to the UrbanCode Velocity plugins page (Settings>Integrations>Plugins) and click “Add Integration” for the Jira plugin. You will be prompted to fill out the “Add Jira Integration” form.
 





The last field defines the Jira projects of interest. Use the Jira project code, not the full name. 


Now that the integration has been added to Velocity, it can also be added to a specific value stream. UrbanCode Velocity value streams are fully configurable by downloading and uploading a value stream map (VSM) which is a specially crafted .json file. Save the vsm.json content provided below as a .json file. We wil then upload this file to configure our value stream.
Because this json content references your Jira integration by name, the integration must be named “JKE Jira 1” and must be “online” before uploading it.
{
  "tenantId": "5ade13625558f2c6688d15ce",
  "integrations": [
    {
      "name":"JKE Jira 1"
    }
  ],
  "phases": [
    {
      "name": "Planning",
      "stages": [
        {
          "name": "Backlog",
          "query": "issue.status=Backlog"
        },
        {
          "name": "Selected For Development",
          "query": "issue.status='Selected for Development' AND (pr.status!=open AND pr.status!=closed)"
        }
      ]
    },
    {
      "name": "Development",
      "stages": [
        {
          "name": "In Progress",
          "query": "(issue.status='In Progress' OR pr.status=open) AND issue.status!='In Review'"
        },
        {
          "name": "In Review",
          "query": "issue.status='In Review' AND pr.status=open"
        },
        {
          "name": "Merged",
          "query": "issue.status=Merged OR pr.status=closed AND build.status!=success"
        },
        {
          "name": "Build",
          "query": "build.status=success AND deployment.env!=DEV"
        }
      ]
    },
    {
      "name": "Deployment",
      "stages": [
        {
          "name": "DEV",
          "query": "deployment.env=DEV AND deployment.env!=QA"
        },
        {
          "name": "QA",
          "query": "deployment.env=QA AND deployment.env!=PROD"
        },
        {
          "name": "PROD",
          "query": "deployment.env=PROD"
        }
      ]
    }
  ]
}
Explanation of VSM JSON
- Integrations Array
This vsm.json content defines an integration for Jira. Integrations are included based on integration name. This workbook uses the name “JKE Jira 1”. This needs changed if the integration happens to have been named differently."integrations":[ { "name":"JKE Jira 1" } ]- Phases and Stages
Value streams are organized as phases and stages. The json file below provides phase and stage definitions for this workbook. An important part of stages are stage queries which logically define whether a work item (dot) should be included in a stage or not."phases":[ { "name": "Planning", "stages": [ { "name": "Backlog", "query": "issue.status=Backlog" }, ...
If you have a brand-new value stream, then the upload button will be directly available.

After the first vsm.json file is uploaded, the value stream can be modified with additional uploads via the tools and utilities dropdown option “Replace value stream map”.

Wait for Velocity to synchronize data. The Jira issue that was placed in the Backlog should now appear in the value stream Backlog.

After setting up the Jira integration in Velocity and using the vsm.json file to add the integration and stage queries to the value stream, the Jira card should be visible as a dot within the value stream. The issue status can be updated in Jira and the work item (dot) will change stages in Velocity.
Move the Jira card from “Backlog” to “Selected for Development”.  

Allow time for Velocity to synchronize (usually ~1 minute max). The work item should moved to next stage “Selected For Development”.     

Click on the dot to view details such as a link back to the Jira card as well as work item history.     

This section has focussed on the planning phase of our value stream. In this case, we used Jira as our issue tracker. We created a Jira board and issue, setup a Jira integration, added that integration to a value stream, and observed how the value stream tracked Jira status changes throughout the planning phase. In theory, we could use Jira for every stage of our value stream; however, in practice, Jira cards are encumbered by manual updates and ulterior constraints. We can do better than that! We can directly integrate with our other systems like GitHub and Jenkins to gain a real-time, fully automated, and accurate visualization of our work item’s journey. Next up is our development phase where we will add a GitHub integration and see how Jira and GitHub can work in tandem.
Questions or suggestions? Create an issue at https://github.com/urbancode/velocity-info/issues
GitHub requirements:
A very easy option is to use a public GitHub account and public repository. The repository contents do not matter, but a PR is required and must contain the issue ID of the Jira issue we are following in the PR name (the example here is “JKET-1”). The steps below are for creating a new PR.
Edit any file in the repository. This example just edits a README file that can be edited by clicking on the pencil icon to the upper right of the file. 
 
Make changes to the file. Think of these changes as corresponding to the Jira card created earlier. This represents the code changes for that card. 
 
Commit the changes to a separate branch. 
 
When prompted to create a pull request, provide the Jira code in the pull request name (ex. “JKET-1”), then click “Create pull request”.                       
  
 
Confirm that the pull request is open and contains the Jira issue ID.
 






Use the vsm.json file to add the GitHub integration and a linkRule to the value stream. Download the file and then modify it by adding the following sections, then upload the file again to apply changes. The work item (dot) should move to the “In Progress” stage since the PR is open.

Add the GitHub integration to the .json content.
Similar to the json that added the Jira integration, we will need to add another json object to the integrations array. The name used here must match the GitHub integration name created earlier. Note that the sample code below shows the “JKE Jira 1” object as only having the name property but a downloaded vsm.json will contain additional details for this object. We only need to add the GitHub object with its single name property for now and can ignore the details for the Jira object.
   "integrations": [
     {
       "name":"JKE Jira 1"
     },
     {
       "name":"JKE GitHub App1"
     }
   ],
Add the linkRules to the .json content.
replace the empty link rules array…
 "linkRules": [],
with an array that contains a new link rule object. This linkRule links GitHub PRs to Jira issues based on a regex pattern that recognizes a jira.id within a pr.name.
   "linkRules": [
     {
       "fromIntegrationName": "JKE GitHub App1",
       "toIntegrationName": "JKE Jira 1",
       "fromField": "pr.name",
       "toField": "issue.id",
       "pattern": "([A-Z]+-[0-9]+)"
     }
   ],






So far we have already gained value by observing our value stream across Jira and GitHub. Next we will see how Jenkins can integrate with UrbanCode Velocity’s pipeline and value stream forming a powerful combination of build and deployment control and tracking.
Questions or suggestions? Create an issue at https://github.com/urbancode/velocity-info/issues
If you do not have a Jenkins server available, See Jenkins documentation for running a local instance https://jenkins.io/doc/book/installing/.
This example uses the pipeline plugin for Jenkins. If you have not already, add this plugin now: https://plugins.jenkins.io/workflow-aggregator.
UrbanCode Velocity value streams can include build and deployment data from Jenkins. Due to the nature of Jenkins being a build server, the setup for it is slightly different than the other plugins discussed in this workbook. For one, because Velocity has the ability to start Jenkins jobs, it requires a plugin on the Jenkins server as well. Also worth noting for when we go to use our integration with value streams is that, because Jenkins is incorporated at the deployment plan and pipeline level of Velocity, it is not added to value streams via the vsm.json.
Install the UrbanCode Velocity plugin on your Jenkins instance.
Navigate to the plugins page on your Jenkins instance by clicking Manage Jenkins > Manage Plugins > Available (tab) and search for UrbanCode Velocity Plugin.  When located install the plugin and restart your instance when possible.
Navigate to the Settings Page of UrbanCode Velocity and select the Integrations section in the left navigation, then click on “Plugins”. Create a new Jenkins integration by clicking “Add Integration”.

Provide a name for your Jenkins integration (should describe the jenkins instance). You can name this integration anything for the workbook. 
Click “Create”. This will generate an Integration ID and Integration Token pair. Click “Copy to Clipboard” to copy these fields and save them. These values will be needed to configure the UrbanCode Velocity plugin installed on Jenkins.
Back in Jenkins, navigate to the Jenkins configuration page Manage Jenkins > Configure System > UrbanCode Velocity (section).  Under the UrbanCode Velocity section paste the Integration ID and Integration Token values generated by the integration we created in Velocity.
Also in Jenkins, Provide credentials for a Jenkins user on whose behalf this plugin may access your Jenkins items.
Depending on your version, you might need to provide a Velocity user access token to the Velocity plugin for Jenkins (the access token field is not shown in the screenshot below). This access token is created within Velocity and should be used uniquely by this Jenkins integration as its own authentication.
Click Apply and save before clicking the Test Connection button to confirm your connection to UrbanCode Velocity. Upon successful connection, your data will be posted to UrbanCode Velocity. 
Jenkins integrations and jobs are made available to the value stream in a different way than other integrations. There is no need to edit the vsm.json file; however, we will need to create an “Application” on the value stream’s pipeline in order to have a target for build and deployment data.




Jenkins publishes data to Velocity using API attributes to identify constructs such as pipelines and applications. We will need these attributes as we configure our Jenkins jobs. You can download API Attributes directly from the value stream under the tools and utilities dropdown > “Download attributes for API”.

Create a new Jenkins Pipeline Job.  
 
 
Copy and paste the script below as a pipeline script. You will need to provide the variables summarized in the table below.
| Variable Name | Description | Example | 
|---|---|---|
| GITHUB_REPO_URL | The URL to your GitHub repository that you are using for this workbook. | https://github.com/UrbanCodeVelocity/JKE-App-1 | 
| GITHUB_BRANCH | The GitHub branch name used for the build. Usually “main” or “master” | main | 
| VELOCITY_ENV_ID_DEV | An ID that uniquely identifies your value stream’s DEV environment | cb348f56-29f3-4ade-9c9f-38daedf3b663 | 
| VELOCITY_ENV_ID_QA | An ID that uniquely identifies your value stream’s QA environment | 7a115f90-f4e5-4181-9920-78b216bb4afc | 
| VELOCITY_APP_NAME | The Velocity pipeline application name (use “JKE App1” for the workbook, we will create this pipeline application later) | JKE App1 | 
node {
    //URL to Github repository https://github.com/<owner>/<repo>
    def GITHUB_REPO_URL="https://github.com/<OWNER>/<REPO NAME>"
    def GITHUB_BRANCH="main"
    
    //Retrieve ENV ID values for DEV and QA from "Download attributes from API" json file
    def VELOCITY_ENV_ID_DEV="<DEV ENVIRONMENT ID>"
    def VELOCITY_ENV_ID_QA="<QA ENVIRONMENT ID>"
    //VELOCITY_APP_NAME must match your Velocity pipeline application name
    def VELOCITY_APP_NAME="<VELOCITY PIPELINE APPLICATION NAME>"
    //Do not change below this line.
    def GIT_COMMIT
    
    stage ('cloning the repository'){
      currentBuild.displayName = "2.1.1.${BUILD_NUMBER}"
      majorVersion="${BUILD_NUMBER}"
      def scm = git branch: "${GITHUB_BRANCH}", url: "${GITHUB_REPO_URL}"
      GIT_COMMIT = sh(returnStdout: true, script: "git rev-parse HEAD").trim()
      echo "GIT_COMMIT=${GIT_COMMIT}"
    }
    stage ("Build") {
        echo "Building ${VELOCITY_APP_NAME} (Build:${currentBuild.displayName}, GIT_COMMIT:${GIT_COMMIT})"
        step($class: 'UploadBuild', 
           tenantId: "5ade13625558f2c6688d15ce",
           revision: "${GIT_COMMIT}",
           appName: "${VELOCITY_APP_NAME}",
           versionName:"${currentBuild.displayName}",
           requestor: "admin", id: "${currentBuild.displayName}"
        )
    }
   stage ("Deploy to DEV") {
    sleep 10
    step([$class: 'UploadDeployment',
          tenantId: "5ade13625558f2c6688d15ce",
          versionName: "${currentBuild.displayName}",
          versionExtId: "${currentBuild.displayName}",
          type: 'Jenkins',
          environmentId: "${VELOCITY_ENV_ID_DEV}",
          environmentName: 'DEV',
          appName: "${VELOCITY_APP_NAME}",
          description: '[Description ex: Terraform Deployment]',
          initiator: "admin",
		  result: 'true'
      ])
   }
   stage ("Deploy to QA") {
    sleep 10
    step([$class: 'UploadDeployment',
          tenantId: "5ade13625558f2c6688d15ce",
          versionName: "${currentBuild.displayName}",
          versionExtId: "${currentBuild.displayName}",
          type: 'Jenkins',
          environmentId: "${VELOCITY_ENV_ID_QA}",
          environmentName: 'QA',
          appName: "${VELOCITY_APP_NAME}",
          description: '[Description ex: Terraform Deployment]',
          initiator: "admin",
		  result: 'true'
      ])
   }
}
Run the job from Jenkins and observe stage changes in your UrbanCode Velocity value stream.
At this point, back in our value stream, there should still be a dot in the “Merged” stage. We will run the Jenkins pipeline job we created which includes three steps: build, DEV deployment, and QA deployment. This means that, upon running the job, we should see the dot make three transitions: Merged –> Build –> DEV –> QA.
Merged -> Build
 
Build -> DEV
 
DEV -> QA
 
Explanation
For the inquisitive reader, the Jenkins script achieves this functionality as follows:
- Build Stage: The
UploadBuildclass is used to upload build data to UrbanCode Velocity. Therevisionparameter is important for linking the build to the work item via GitHub data (GIT_COMMITin this case). TheversionNameis important for linking forward to deployments. TheappNamecorresponds to the UrbanCode Velocity pipeline application name.step($class: 'UploadBuild', tenantId: "5ade13625558f2c6688d15ce", revision: "${GIT_COMMIT}", appName: "${VELOCITY_APP_NAME}", versionName:"${currentBuild.displayName}", requestor: "admin", id: "${currentBuild.displayName}" )- Deployments (DEV and QA): The
UploadDeploymentclass is used to upload deployment data to UrbanCode Velocity. TheversionNameparameter is critical for linking to build data. TheappNamecorresponds to the UrbanCode Velocity pipeline application name, whileenvironmentNameandenvironmentIdare used to identify the deployment environment.stage ("Deploy to DEV") { sleep 20 step([$class: 'UploadDeployment', tenantId: "5ade13625558f2c6688d15ce", versionName: "${currentBuild.displayName}", versionExtId: "${currentBuild.displayName}", type: 'Jenkins', environmentId: "${VELOCITY_ENV_ID_DEV}", environmentName: 'DEV', appName: "${VELOCITY_APP_NAME}", description: '[Description ex: Terraform Deployment]', initiator: "admin", result: 'true' ]) } stage ("Deploy to QA") { sleep 20 step([$class: 'UploadDeployment', tenantId: "5ade13625558f2c6688d15ce", versionName: "${currentBuild.displayName}", versionExtId: "${currentBuild.displayName}", type: 'Jenkins', environmentId: "${VELOCITY_ENV_ID_QA}", environmentName: 'QA', appName: "${VELOCITY_APP_NAME}", description: '[Description ex: Terraform Deployment]', initiator: "admin", result: 'true' ]) }
StartTime and endTime: The example here does not pass startTime or endTime to either build or deployment uploads. If the startTime and endTime are not provided, then the Jenkins plugin defaults to the current time for both. For long running builds and deployments, a script can upload startTime and endTime values as needed.
Create a second pipeline job, this time for deployment to our production environment.
 
 

Make the pipeline job parameterized with a string parameter named “buildNumber”. 

Copy and paste the script below as a pipeline script. The table below summarizes the variables you will need to provide within the script.
| Variable Name | Description | Example | 
|---|---|---|
| VELOCITY_ENV_ID_PROD | An ID that uniquely identifies your value stream’s PROD environment | 7a115f90-f4e5-4181-9920-78b216bb4afc | 
| VELOCITY_APP_NAME | The Velocity pipeline application name (use “JKE App1” for the workbook, we will create this pipeline application later) | JKE App1 | 
parameters([
    string(name: 'buildNumber', description: 'The version of the application to deploy.')
])
node {
  //Get value for VELOCITY_ENV_ID_PROD from "Download attributes from API" json file.
  def VELOCITY_ENV_ID_PROD="<PROD ENVIRONMENT ID>"
  //VELOCITY_APP_NAME must match your Velocity pipeline application name
  def VELOCITY_APP_NAME="<VELOCITY PIPELINE APPLICATION NAME>"
 
 currentBuild.displayName = "${buildNumber}"
    stage ("Deploy to PROD") {
        step([$class: 'UploadDeployment',
            tenantId: "5ade13625558f2c6688d15ce",
            versionName: "${currentBuild.displayName}",
            versionExtId: "${currentBuild.displayName}",
            type: 'Jenkins',
            environmentId: "${VELOCITY_ENV_ID_PROD}",
            environmentName: 'PROD',
            appName: "${VELOCITY_APP_NAME}", 
            description: '[Description ex: Terraform Deployment]',
            initiator: "admin",
            result: 'true' 
        ])
    }
}
Apply and Save to save the pipeline job.
 



buildNumber as a parameter to a Jenkins job. Since we configured our job with parameters, we will be prompted at this point to provide them. Velocity has various version/inventory parameters on hand to support a fluid pipeline. In this case we’ll enter ${version.buildNumber} into the form.  
 After Running Jenkins Job
  After Running Jenkins Job 




We have successfully navigated an entire value stream from backlog to production deployment spread across four separate tools and systems: three external tools (Jira, GitHub, and Jenkins), and Velocity itself as a deployment tool. We learned about key UrbanCode Velocity concepts such as the vsm.json file with its stage queries, integrations, and linkRules, as well as pipelines and deployments. From here you can go back and begin experimenting with your value stream. One good place to start is by looking at the vsm.json file stages and queries. From there you can begin to shape the value stream to match your own processes.
Questions or suggestions? Create an issue at https://github.com/urbancode/velocity-info/issues