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

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...

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>     ...

Load testing MongoDB using JMeter

Apache JMeter ( http://jmeter.apache.org/ ) added support for MongoDB since its 2.10 release. In this post I am referring to the latest JMeter release (2.13). A preliminary JMeter setup is needed before starting your first test plan for MongoDB. It uses Groovy as scripting reference language, so Groovy needs to be set up for our favorite load testing tool. Follow these steps to complete the set up: Download Groovy from the official website ( http://www.groovy-lang.org/download.html ). In this post I am referring to the Groovy release 2.4.4, but using later versions is fine. Copy the groovy-all-2.4.4.jar to the $JMETER_HOME/lib folder. Restart JMeter if it was running while adding the Groovy JAR file. Now you can start creating a test plan for MongoDB load testing. From the UI select the MongoDB template ( File -> Templates... ). The new test plan has a MongoDB Source Config element. Here you have to setup the connection details for the database to be tested: The Threa...