持续集成 Jenkins pipeline 用 groovy 模板发送 RF 报告的数据,无法获取到对象;用自由风格的任务没问题

he110c · 2018年08月20日 · 最后由 MitnickEX 回复于 2019年07月18日 · 1505 次阅读

Jenkins pipeline 用 groovy 模板发送 RF 报告的数据,无法获取到对象;
具体如下:

Exception raised during template rendering: Cannot get property 'simpleName' on null object

java.lang.NullPointerException: Cannot get property 'simpleName' on null object
    at org.codehaus.groovy.runtime.NullObject.getProperty(NullObject.java:60)
    at org.codehaus.groovy.runtime.InvokerHelper.getProperty(InvokerHelper.java:174)
    at org.codehaus.groovy.runtime.callsite.NullCallSite.getProperty(NullCallSite.java:47)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.callGetProperty(AbstractCallSite.java:296)
    at SimpleTemplateScript1$_run_closure1.doCall(SimpleTemplateScript1.groovy:46)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:497)
    at org.codehaus.groovy.reflection.CachedMethod.invoke(CachedMethod.java:93)
    at groovy.lang.MetaMethod.doMethodInvoke(MetaMethod.java:325)
    at org.codehaus.groovy.runtime.metaclass.ClosureMetaClass.invokeMethod(ClosureMetaClass.java:294)
    at groovy.lang.MetaClassImpl.invokeMethod(MetaClassImpl.java:1022)
    at groovy.lang.Closure.call(Closure.java:414)
    at groovy.lang.Closure.call(Closure.java:430)
    at org.codehaus.groovy.runtime.DefaultGroovyMethods.each(DefaultGroovyMethods.java:2040)
    at org.codehaus.groovy.runtime.DefaultGroovyMethods.each(DefaultGroovyMethods.java:2025)
    at org.codehaus.groovy.runtime.DefaultGroovyMethods.each(DefaultGroovyMethods.java:2066)
    at org.codehaus.groovy.runtime.dgm$162.invoke(Unknown Source)
    at org.codehaus.groovy.runtime.callsite.PojoMetaMethodSite$PojoMetaMethodSiteNoUnwrapNoCoerce.invoke(PojoMetaMethodSite.java:274)
    at org.codehaus.groovy.runtime.callsite.PojoMetaMethodSite.call(PojoMetaMethodSite.java:56)
    at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:48)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:113)
    at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:125)
    at SimpleTemplateScript1.run(SimpleTemplateScript1.groovy:45)
        。。。

def actions = build.actions // List
actions.each() { action ->
if( action.class.simpleName.equals("RobotBuildAction") ) { // hudson.plugins.robot.RobotBuildAction
robotResults = true
应该是这边的问题,但是不知道怎么调试😂
有人遇到过么,求指导

模板文件是网上的,大都类似:

<%
import java.text.DateFormat
import java.text.SimpleDateFormat
%>
<STYLE>
BODY, TABLE, TD, TH, P {
  font-family:Verdana,Helvetica,sans serif;
  font-size:11px;
  color:black;
}
h1 { color:black; }
h2 { color:black; }
h3 { color:black; }
TD.bg1 { color:white; background-color:#0000C0; font-size:120% }
TD.bg2 { color:white; background-color:#4040FF; font-size:110% }
TD.bg3 { color:white; background-color:#8080FF; }
TD.test_passed { color:blue; }
TD.test_failed { color:red; }
TD.console { font-family:Courier New; }
</STYLE>
<BODY>

<TABLE>
  <TR><TD align="right"><IMG SRC="${rooturl}<%= build.result == hudson.model.Result.SUCCESS  ? "static/e59dfe28/images/32x32/blue.gif" : "static/e59dfe28/images/32x32/red.gif" %>" />
  </TD><TD valign="center"><B style="font-size: 150%;"><%= build.result == hudson.model.Result.SUCCESS ? "TESTSRUN ${build.result}" : "TESTRUN ${build.result}" %></B></TD></TR>
  <TR><TD>Project:</TD><TD>${project.name}</TD></TR>
  <TR><TD>Describe:</TD><TD><b>${project.description}</b></TD></TR>
  <TR><TD>Version:</TD><TD><b>${build.environment.version}<b></TD></TR>
  <TR><TD>Number:</TD><TD>${build.displayName}</TD></TR>
  <TR><TD>Date of run:</TD><TD>${it.timestampString}</TD></TR>
  <TR><TD>Test duration:</TD><TD>${build.durationString}</TD></TR>
  <TR><TD>Build URL:</TD><TD><A href="${rooturl}${build.url}">${rooturl}${build.url}</A></TD></TR>
  <TR><TD>Test report:</TD><TD><A href="http://jenkins.corp.mama.cn/jenkins/job/${project.name}/${build.number}/robot/report/report.html">Open report.html</A></TD></TR>
  <TR><TD>Test log:</TD><TD><A href="http://jenkins.corp.mama.cn/jenkins/job/${project.name}/${build.number}/robot/report/log.html">Open log.html</A></TD></TR>
  <TR><TD>Console logs:</TD><TD><A href="${rooturl}${build.url}console">Console</A></TD></TR>
</TABLE>
<BR/>

<!-- Robot Framework Results -->
<%
def robotResults = false
def actions = build.actions // List<hudson.model.Action>
actions.each() { action ->
  if( action.class.simpleName.equals("RobotBuildAction") ) { // hudson.plugins.robot.RobotBuildAction
    robotResults = true %>
<p><h4>Robot Framework Results</h4></p>
<table cellspacing="0" cellpadding="4" border="1" align="left">
        <thead>
          <tr bgcolor="#F3F3F3">
            <td><b>Type</b></td>
            <td><b>Total</b></td>
            <td><b>Passed</b></td>
            <td><b>Failed</b></td>
            <td><b>Pass %</b></td>
          </tr>
        </thead>

        <tbody>

          <tr><td><b>All Tests</b></td>
            <td><%= action.result.overallTotal %></td>
            <td><%= action.result.overallPassed %></td>
            <td><%= action.result.overallFailed %></td>
            <td><%= action.overallPassPercentage %></td>
          </tr>

          <tr><td><b>Critical Tests</b></td>
            <td><%= action.result.criticalTotal %></td>
            <td><%= action.result.criticalPassed %></td>
            <td><%= action.result.criticalFailed %></td>
            <td><%= action.criticalPassPercentage %></td>
          </tr>

        </tbody>
      </table>
<br />
<br />
<br />
<br />
<br />
<br />
<table cellspacing="0" cellpadding="4" border="1" align="left">
<thead>
<tr bgcolor="#F3F3F3">
  <td><b>Test Name</b></td>
  <td><b>Status</b></td>
  <td><b>Execution Datetime</b></td>
</tr>
</thead>
<tbody>
<%  def suites = action.result.allSuites
    suites.each() { suite -> 
      def currSuite = suite
      def suiteName = currSuite.displayName
      // ignore top 2 elements in the structure as they are placeholders
      while (currSuite.parent != null && currSuite.parent.parent != null ) {
        currSuite = currSuite.parent
        suiteName = currSuite.displayName + "." + suiteName
      } %>
<tr><td colspan="3"><b><%= suiteName %></b></td></tr>
<%    DateFormat format = new SimpleDateFormat("yyyyMMdd HH:mm:ss.SS")
      def execDateTcPairs = []
      suite.caseResults.each() { tc ->
        Date execDate = format.parse(tc.starttime)
        execDateTcPairs << [execDate, tc]
      }
      // primary sort execDate, secondary displayName
      execDateTcPairs = execDateTcPairs.sort{ a,b -> a[1].displayName <=> b[1].displayName }
      execDateTcPairs = execDateTcPairs.sort{ a,b -> a[0] <=> b[0] }
      execDateTcPairs.each() {
        def execDate = it[0]
        def tc = it[1]  
        def tcresult = tc.isPassed() %>
<tr>
  <td><%= tc.displayName%></td>
  <td style="color: <%= tc.isPassed() ? "#66CC00" : "#FF3333" %>"><%= tc.isPassed() ? "PASS" : "FAIL" %></td>
  <td><%= execDate %></td>
</tr>
<%    } // tests
    } // suites %>
</tbody></table><%
  } // robot results
}
if (!robotResults) { %>
<p>No Robot Framework test results found.</p>
<%
} %>
</BODY>
共收到 3 条回复 时间 点赞

是不是没装插件 RobotBuildAction?

将 if( action.class.simpleName.equals("RobotBuildAction") )
替换为
if( action instanceof hudson.plugins.robot.RobotBuildAction )
可以至少可以完成

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册