Skip to main content

How to check plugins usage in Jenkins

It's a good practice for a Jenkins administrator to check for installed plugins usage before updating one of them or to understand if some isn't used any more and can be uninstalled. The Enterprise Jenkins release provides a plugin (http://www.cloudbees.com/jenkins-enterprise-by-cloudbees-features-plugin-usage-plugin.cb) that shows in a tabular form the usage of the installed plugins across the different jobs in a Jenkins instance. It is a really simple but extremely helpful plugin. But how could you do the same if no Enterprise Jenkins is available for you? The same functionality can be implemented with a quite simple Groovy script using the Jenkins classes only. This script can be executed from the Jenkins script console (accessible from Manage Jenkins -> Script Console). Here's the code explained step by step. For a better understanding of this post I suggest you have the basics about the Jenkins model and the Jenkins Java classes.
First of all we need to get all the current instances of hudson.model.Project:

// Get all the current projects for this Jenkins instance
def currentProjects = Hudson.instance.items.findAll{ it instanceof Project }

Then we cycle among them. For each item we get its Descriptor. If it is the first occurence of a descriptor we add it as a new key to a Map. If not, simply associate the project to the plugin Descriptor key in the Map:

// Define the project grouped by descriptor Map
def projectsGroupedByDescriptor = [:]
 
// Cycle between projects
for(def project in currentProjects) {
    def items = new ArrayList(project.publishers.values())
    items.addAll(project.builders)
    // Cycle each item of the current project
    for(def item in items) {
         // Get the item Descriptor
        def descriptor = item.descriptor
        // If this is the first occurrence, put a new key into the Map
        if (!projectsGroupedByDescriptor.containsKey(descriptor)) {
            projectsGroupedByDescriptor.put(descriptor, [])
        }
        // Check, through the plugin descriptor, if the project uses the plugin
        if (!projectsGroupedByDescriptor[descriptor].contains(project)) {
            // Associate the project to the plugin descriptor key in the Map
            projectsGroupedByDescriptor[descriptor].add(project)
        }
    }   
}


This way we got all the content we need. Now we need to choose a way to render it. The following code is one simple proposal, but not the only one possible. For each plugin it prints its name, description and the Builder full class name (this is important because sometimes plugin developers forget to overwrite the getDisplayName() method of the plugin Descriptor, so you could see a bunch of Hello World plugin names in the final list and could be hard to recognize some of them) and then the list and the count of the jobs using it. The plugin list is ascending ordered by usage. The result is printed in the script console page:

// Define a StringBuffer for the final output
def output = ''<<''
// Format the output using the Map
for (def descriptor in projectsGroupedByDescriptor.keySet().sort { projectsGroupedByDescriptor.get(it).size() }) {
  output << descriptor.displayName
  output << '\n'
  output << (descriptor.plugin?.displayName?:'Jenkins Core')
  output << '\n'
  if(descriptor.class.name.lastIndexOf('$') != -1) {
      output << descriptor.class.name.substring(0, descriptor.class.name.lastIndexOf('$'))
  } else {
      output << descriptor.class.name
  }
  output << '\n'
  def projectCount = 0
  for (def project in projectsGroupedByDescriptor.get(descriptor)) {
    projectCount++
    output << '\t'
    output << project.fullDisplayName
    output << '\n'
  }
  output << 'Plugin usage: ' + projectCount
  output << '\n'
  output << '--------------------------------'
  output << '\n'
}


The output is something like this:

My Plugin 1
This is my first custom plugin for Jenkins
ie.googlielmo.MyFirstPlugin
    job 2
    job 3
Plugin usage: 2
--------------------------------
My Plugin 2
This is my second custom plugin for Jenkins
ie.googlielmo.MySecondPlugin
    job 1
    job 4
    job 5
Plugin usage: 3
--------------------------------
...


The code presented could be executed also in a System Groovy Script build step of a build job using the Groovy Plugin (https://wiki.jenkins-ci.org/display/JENKINS/Groovy+plugin).
This is a way to check the plugins usage safest than browsing across the XML desciptors of the single jobs (this is the most common solution I found in the web for this matter). Of course some improvement to the code above could be added and you can implement a better rendering exporting the results to a different format (XML, HTML, PDF, Json, etc.) but the goal of this post was to show how it is possible to implement an helpful functionality for Jenkins in a easy way simply using the available APIs. The solution proposed could be also used as a base suggestion to implement a custom plugin (if you really need it) to implement this check functionality.

Comments

Popular posts from this blog

Exporting InfluxDB data to a CVS file

Sometimes you would need to export a sample of the data from an InfluxDB table to a CSV file (for example to allow a data scientist to do some offline analysis using a tool like Jupyter, Zeppelin or Spark Notebook). It is possible to perform this operation through the influx command line client. This is the general syntax: sudo /usr/bin/influx -database '<database_name>' -host '<hostname>' -username '<username>'  -password '<password>' -execute 'select_statement' -format '<format>' > <file_path>/<file_name>.csv where the format could be csv , json or column . Example: sudo /usr/bin/influx -database 'telegraf' -host 'localhost' -username 'admin'  -password '123456789' -execute 'select * from mem' -format 'csv' > /home/googlielmo/influxdb-export/mem-export.csv

jOOQ: code generation in Eclipse

jOOQ allows code generation from a database schema through ANT tasks, Maven and shell command tools. But if you're working with Eclipse it's easier to create a new Run Configuration to perform this operation. First of all you have to write the usual XML configuration file for the code generation starting from the database: <?xml version="1.0" encoding="UTF-8" standalone="yes"?> <configuration xmlns="http://www.jooq.org/xsd/jooq-codegen-2.0.4.xsd">   <jdbc>     <driver>oracle.jdbc.driver.OracleDriver</driver>     <url>jdbc:oracle:thin:@dbhost:1700:DBSID</url>     <user>DB_FTRS</user>     <password>password</password>   </jdbc>   <generator>     <name>org.jooq.util.DefaultGenerator</name>     <database>       <name>org.jooq.util.oracle.OracleDatabase</name>     ...

Turning Python Scripts into Working Web Apps Quickly with Streamlit

 I just realized that I am using Streamlit since almost one year now, posted about in Twitter or LinkedIn several times, but never wrote a blog post about it before. Communication in Data Science and Machine Learning is the key. Being able to showcase work in progress and share results with the business makes the difference. Verbal and non-verbal communication skills are important. Having some tool that could support you in this kind of conversation with a mixed audience that couldn't have a technical background or would like to hear in terms of results and business value would be of great help. I found that Streamlit fits well this scenario. Streamlit is an Open Source (Apache License 2.0) Python framework that turns data or ML scripts into shareable web apps in minutes (no kidding). Python only: no front‑end experience required. To start with Streamlit, just install it through pip (it is available in Anaconda too): pip install streamlit and you are ready to execute the working de...