接触Jenkins(Hudson)API,第1部分

哪一个-哈德森还是詹金斯?

都。 几个月前,我开始使用Hudson v1.395来从事这个小项目,在出现巨大分歧之后又回到了这个项目。 我以此为契机,看我将来选择永久搬到詹金斯时是否会遇到任何重大问题。 有很多麻烦-最值得注意的是,新的CLI jar不能立即使用-但Jenkins的整体v1.401在切换后按预期工作。 好消息是,旧版本的CLI jar仍然有效,因此此示例实际上是使用代码混合来完成工作。 无论如何,该软件非常出色,并且值得称赞的有余。

API

Jenkins / Hudson有一个便捷的远程API,其中包含有关您的构建的信息,并支持一组丰富的功能来远程控制它们以及服务器(通常是服务器)。 可以触发构建,复制作业,停止服务器,甚至远程安装插件。 与服务器的API交互时,可以选择XML,JSON或Python。 而且,正如内置文档所述,您可以在以下位置的相对路径中找到构建服务器URL所需的功能:

“ /.../api/,其中“ ...”部分是您要访问的对象”。

如果您在浏览器中导航到该页面,它将显示一个简短的文档页面,如果您将所需的格式添加为路径的最后一部分,则将返回结果。 例如,要加载有关运行本地托管Jenkins服务器的计算机的信息,对此URL的get请求将以JSON格式返回结果:http:// localhost:8080 / computer / api / json。

{'busyExecutors': 0,'displayName': 'nodes','computer': [{'idle': true,'executors': [{},{}],'actions': [],'temporarilyOffline': false,'loadStatistics': {},'displayName': 'master','oneOffExecutors': [],'manualLaunchAllowed': true,'offline': false,'launchSupported': true,'icon': 'computer.png','monitorData': {'hudson.node_monitors.ResponseTimeMonitor': {'average': 111},'hudson.node_monitors.ClockMonitor': {'diff': 0},'hudson.node_monitors.TemporarySpaceMonitor': {'size': 58392846336},'hudson.node_monitors.SwapSpaceMonitor': null,'hudson.node_monitors.DiskSpaceMonitor': {'size': 58392846336},'hudson.node_monitors.ArchitectureMonitor': 'Mac OS X (x86_64)'},'offlineCause': null,'numExecutors': 2,'jnlpAgent': false}],'totalExecutors': 2
}

这是使用GraphViz渲染的同一棵树。

此功能从服务器的根开始在树中扩展,您可以通过在网址上提供“ depth”参数来控制从任何特定分支加载的树的数量。 请注意您指定此变量的高度。 在人口众多,运行时间较长的构建服务器(数十个具有数千个作业执行的构建)上进行了四个加载深度的测试,从而设法使我定期超时。 为了让您有个想法,这里是api根深3处的域的非常粗略的可视化。

从服务器中获取数据非常简单,但是远程触发服务器上的活动的能力更加有趣。 为了触发名为“ test”的作业的构建,http:// localhost:8080 / job / test / build上的POST执行了该作业。 使用可用的设施,很容易做到:

  • 加载作业的配置文件,对其进行修改并通过发布新的config.xml文件来创建新作业
  • 将作业从一台构建机器移至另一台
  • 建立计划的构建概述

CLI Jar

还有另一种方法可以在与服务器一起分发的CLI jar中远程驱动构建服务器。 这个jar提供了用于在构建服务器上远程执行某些命令的简单工具。 值得注意的是,这使远程安装插件和执行远程Groovy Shell成为可能。 我将这个功能与CLI罐公开的主类的非常薄的包装器结合在一起,如下一个代码示例所示。

/*** Drive the CLI with multiple arguments to execute.* Optionally accepts streams for input, output and err, all of which* are set by default to System unless otherwise specified.* @param rootUrl* @param args* @param input* @param output* @param err* @return*/
def runCliCommand(String rootUrl, List<String> args, InputStream input = System.in,OutputStream output = System.out, OutputStream err = System.err)
{def CLI cli = new CLI(rootUrl.toURI().toURL())cli.execute(args, input, output, err)cli.close()
}

这是一个简单的测试,显示了如何执行Groovy脚本以加载有关作业的信息,类似于您可以从服务器上内置的Groovy脚本控制台执行的操作,该操作可在http:/本地安装的部署中找到。 / localhost:8080 / script。

def 'should be able to query hudson object through a groovy script'()
{final ByteArrayOutputStream output = new ByteArrayOutputStream()when:api.runCliCommand(rootUrl, ['groovysh', 'for(item in hudson.model.Hudson.instance.items) { println('job $item.name')}'],System.in, output, System.err)then:println output.toString()output.toString().split('\n')[0].startsWith('job')
}

如果您想了解更多信息,下面是一些有关CLI的文章链接:

  • 哈德逊CLI Wikidoc
  • Jenkins CLI Wikidoc
  • Jenkins上PHP作业的模板
  • 川口浩辅的文章
  • 一个不错的教程

HTTPBuilder

如今 ,当针对HTTP API进行编程时, HTTPBuilder是我的首选工具。 用法非常简单,我只能使用两种方法来支持到达整个API:一种用于GET,一种用于POST。 这是GET方法,足以执行请求,解析JSON响应并完成(尽管很幼稚)错误处理。

/*** Load info from a particular rootUrl+path, optionally specifying a 'depth' query* parameter(default depth = 0)** @param rootUrl the base url to access* @param path  the api path to append to the rootUrl* @param depth the depth query parameter to send to the api, defaults to 0* @return parsed json(as a map) or xml(as GPathResult)*/
def get(String rootUrl, String path, int depth = 0)
{def statusHTTPBuilder http = new HTTPBuilder(rootUrl)http.handler.failure = { resp ->println 'Unexpected failure on $rootUrl$path: ${resp.statusLine} ${resp.status}'status = resp.status}def infohttp.get(path: path, query: [depth: depth]) { resp, json ->info = jsonstatus = resp.status}info ?: status
}

调用它来获取数据是一个内衬,因为唯一的真正区别是调用API时使用的“路径”变量。

private final GetRequestSupport requestSupport = new GetRequestSupport()...
/*** Display the job api for a particular Hudson job.* @param rootUrl the url for a particular build* @return job info in json format*/
def inspectJob(String rootUrl, int depth = 0)
{requestSupport.get(rootUrl, API_JSON, depth)
}

从技术上讲,这里没有什么可以将其限制为仅JSON。 HTTPBuilder的一大优点是,它将很高兴地尝试对响应进行正确的处理。 如果返回的数据为JSON格式(如以下示例所示),则将其解析为JSONObject。 另一方面,如果数据是XML,则将其解析为Groovy GPathResult。 尽管导航对象图的语法不同,但它们都很容易导航。

你能做什么呢?

探索Hudson / Jenkins API的主要动机是了解如何使管理多台服务器变得更加容易。 目前,我每天处理四台构建服务器和另一台从属计算机,并支持各种不同版本的分支。 这包括单元测试套件和功能测试套件的混合,以及持续进行的部署作业,该作业定期将更改推送到与我们支持的平台矩阵匹配的测试机上,因此,不幸的是,事情并没有分支时复制单个作业那么简单。 确实,以自动或至少半自动的方式为新功能分支创建构建基础结构确实很有吸引力,尤其是因为正在制定扩展构建自动化的计划时。 对于最近的555天项目,我利用API层构建了Grails应用程序,该应用程序既可以作为跨服务器构建的辐射器,又可以用作服务器管理的中央工具。 该概念证明能够连接到多个构建服务器并可视化作业数据以及特定的系统配置,触发构建,并直接链接到每个连接的服务器以允许进一步钻取。 这是几个样机,几乎可以显示图片。

只是一个非常酷的应用程序,用于安装Jenkins

这只是非常间接的关系,但是我遇到了一个非常漂亮且简单的Griffon应用程序,称为Jenkins-Assembler ,它简化了构建服务器的准备工作。 它为您提供了一系列插件,让您选择并选择,然后下载并将它们组合成一个可部署的战争。

足够多的谈话–代码在哪里???

与本文相关的源代码可在github上找到 。 这些测试更多地是对实时API的探索,而不是对该项目中代码的实际测试。 它们在使用Gradle Jetty插件启动的本地服务器上运行。 最后,这是为您准备的一些漂亮图片。
[以幻灯片显示]
[使用PicLens查看]

继续第2部分 。

参考:在The Kaptain on…博客上,从我们的JCG合作伙伴 Kelly Robinson接触到Jenkins(Hudson)API 。


翻译自: https://www.javacodegeeks.com/2012/08/hooking-into-jenkins-hudson-api-part-1.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/372049.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用javascript模拟常见数据结构(四)

七、树 树是一种非线性的分层的数据结构&#xff0c;在现实生活中比较常见的例子比如家谱和公司的组织架构图&#xff0c;如下所示&#xff1a; 一个树结构存在着一系列的父子结构&#xff0c;并且有着一个根节点&#xff0c;这种结构本质上表明了一对多的关系。 那&#xff0c…

最全Pycharm教程(10)——Pycharm调试器总篇

最全Pycharm教程&#xff08;1&#xff09;——定制外观 最全Pycharm教程&#xff08;2&#xff09;——代码风格 最全Pycharm教程&#xff08;3&#xff09;——代码的调试、执行 最全Pycharm教程&#xff08;4&#xff09;——有关Python解释器的相关配置 最全Pycharm教程&am…

Looper.prepare()和Looper.loop()

什么时候需要 Looper Looper用于封装了android线程中的消息循环&#xff0c;默认情况下一个线程是不存在消息循环&#xff08;message loop&#xff09;的&#xff0c;需要调用Looper.prepare()来给线程创建一个消息循环&#xff0c;调用Looper.loop()来使消息循环起作用&#…

如何查看Ubuntu版本,以及Linux内核版本??

查看Ubuntu版本&#xff1a; 方法一&#xff1a; cat /etc/issue 方法二&#xff1a; sudo lsb_release -a 查看内核版本&#xff1a; uname -r 转载于:https://www.cnblogs.com/tanrong/p/6937749.html

实现chrome扩展启动本地进程 - 补充

实现chrome扩展启动本地进程 - 补充 标签&#xff1a; chrome扩展启动本地程序访问本地磁盘2014-10-17 11:42 6753人阅读 评论(17) 收藏 举报分类&#xff1a;Chrome Plugin版权声明&#xff1a;本文为博主原创文章&#xff0c;未经博主允许不得转载。 示例 主要包含如下部分 c…

单路电压表c语言编程,用AT89C51单片机制作的数字电压表

此数字电压表&#xff0c;利用A/D转换原理将被测模拟量转换成数字量&#xff0c;并通过控制系统用数字方式显示测量结果。本设计采用AT89C51单片机&#xff0c;ADC0809进行模/数转换&#xff0c;能够测量8路0&#xff5e;5V的输入电压值&#xff0c;可用四位LED数码管轮流或单路…

ZK的实际应用:MVVM –加载和渲染数据

先前的文章简要介绍了RIA框架ZK&#xff0c;以及它CSS Selector启发式控制器机制如何通过使在控制器类中引用UI组件的任务变得相对灵活来减轻UI更改所带来的一些负担。 然后&#xff0c;我们在上一篇文章中探讨了ZK中的MVVM模式如何允许单个ViewModel提供不同的视图。 这篇文章…

搭建一个简单的mybatis框架

一、Mybatis介绍 MyBatis是一个支持普通SQL查询&#xff0c;存储过程和高级映射的优秀持久层框架。MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装。MyBatis可以使用简单的XML或注解用于配置和原始映射&#xff0c;将接口和Java的POJO&#xff08;Pla…

c语言空格符 r t,c语言中、\t \r \n 和空格什么意思

具体意思&#xff1a;都是转义字符&#xff0c;空格就是单纯的空格&#xff0c;输入时可以输入空格\t 跳格 \r 回车 \n 换行\\ 反斜杠 \a 警告 \b 退格 \f 换页 \v 垂直跳格 \ddd ddd 是 1、2 或 3 位八进制数字。转义字符串(Escap…

在代理类中引用动态代理

在Stackoverflow中有一个有趣的问题 &#xff0c;关于Spring Bean如何获​​得对由Spring创建的代理的引用以处理事务&#xff0c;Spring AOP&#xff0c;缓存&#xff0c;异步流等。需要对代理的引用&#xff0c;因为如果存在对自身的调用通过代理bean&#xff0c;此调用将完全…

android仿高德地图透明黑字,Android 仿高德地图可拉伸的BottomSheet

原标题&#xff1a;Android 仿高德地图可拉伸的BottomSheet2018安卓巴士开发者大会-上海站你一直期待的安卓技术盛宴即将登场&#xff01;前言最近项目中需要用到高德地图搜索结果后的结果展示的可拉伸控件。而我看到这个效果图&#xff0c;觉得这个就是一个slidingpanel&#…

[ Javascript ] JavaScript中的定时器(Timer) 是怎样工作的!

作为入门者来说。了解JavaScript中timer的工作方式是非常重要的。通常它们的表现行为并非那么地直观&#xff0c;而这是由于它们都处在一个单一线程中。让我们先来看一看三个用来创建以及操作timer的函数。var id setTimeout(fn, delay); - 初始化一个单一的timer&#xff0c…

Android 软键盘自动弹出和关闭

在我们写修改信息或者搜索&#xff0c;修改密码等界面的时候&#xff0c;用户进入这个界面的主要目的就是输入修改/查找 某些信息&#xff0c;为了用户体验应该自动弹出软键盘而不是让用户主动点击输入框才弹出。 1.软键盘的自动弹出 private void showKeyboard(){InputMethodM…

android adb杀死服务,Android app是如何杀掉的

1. adb shell kill -9 pid_of_appAMS定义了AppDeathRecipientAPP 在 attachApplication -> attachApplicationLockedAMS里会注册 App 进程的 BinderDeath通知AppDeathRecipient adr new AppDeathRecipient(app, pid, thread);thread.asBinder().linkToDeath(adr, 0);当App进…

iOS学习笔记39-ReactiveCocoa入门

FRP&#xff0c;全称为Functional Reactive Programming&#xff0c;是一种响应变化的编程范式&#xff0c;最近几年比较火&#xff0c;大概的理解就像这样&#xff1a; 当a的值或者b的值发生变化时&#xff0c;c的值会自动响应a的值或b的值变化的信号&#xff0c;自动更正自己…

使用密码摘要生成器扩展JMeter

最近&#xff0c;我不得不处理一个带有50,000条用户记录的OpenLDAP实例&#xff0c;并进行一些压力测试。 JMeter是填充LDAP的最佳选择。 但是&#xff0c;在我的情况下&#xff0c;OpenLDAP配置为不接受任何明文密码。 因此&#xff0c;我无法使用通过JMeter LDAP Request采…

制造业数字化转型核心不止是技术

一、制造业的数字化转型意味着什么&#xff1f; 在当今的制造业领域&#xff0c;数字化转型意味着通过集成数字技术来增强传统的制造方法、产品和劳动力的过程。这些技术包括一系列创新&#xff0c;如自动化软件、电子商务系统、传感器、工业机器人等。 二、制造业数字化转型的…

5分钟内Google App Engine上的Vaadin App

在本教程中&#xff0c;您将学习如何创建第一个Vaadin Web应用程序&#xff0c;如何在本地AppEngine开发服务器上运行它以及如何将其部署到Google App Engine基础结构。 所有这些大约需要5到10分钟。 是的&#xff0c;如果您安装了必要的先决条件&#xff0c;则可以立即开始运行…

android8强制将app移到sd卡,小内存手机 APP强制转移至SD卡教程

虽然近两年手机的机身内存越做越大&#xff0c;但是身边总还是有些朋友在使用几年前的手机。而面对如今海量的丰富应用&#xff0c;早年的手机中内置的存储空间已经开始捉襟见肘。虽说对于这类机型系统通常都提供了将APP转移至外置内存卡的功能&#xff0c;可是依然有一些顽固的…

android 书架菜单,Android入门3--做一个书架

修改名称创建项目的时候&#xff0c;APP的名字取为英文或者拼音&#xff0c;是为了简便&#xff0c;但是显示在界面上&#xff0c;我们当然希望它是中文的。taoguanstring>我们要做的很简单&#xff0c;就是在string.xml中&#xff0c;将app_name的内容修改为我们希望的名字…