{"id":95720,"date":"2020-11-17T19:06:51","date_gmt":"2020-11-17T18:06:51","guid":{"rendered":"https:\/\/blog.jetbrains.com\/?post_type=youtrack&#038;p=95720"},"modified":"2020-11-17T19:07:17","modified_gmt":"2020-11-17T18:07:17","slug":"workflow-part-13","status":"publish","type":"youtrack","link":"https:\/\/blog.jetbrains.com\/zh-hans\/youtrack\/2020\/11\/workflow-part-13\/","title":{"rendered":"\u4f18\u5316\u5de5\u4f5c\u6d41 \u2014 \u7b2c 13 \u90e8\u5206\uff1a\u652f\u6301\u6d4b\u8bd5\u7ba1\u7406\u65b9\u6848"},"content":{"rendered":"<p><a href=\"https:\/\/blog.jetbrains.com\/youtrack\/2020\/10\/make-it-workflow-part-13-supporting-test-management-scenarios\/\/\"><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/YT_workflow_en-1.png\" alt=\"\" \/><\/a><\/p>\n<p>\u6b22\u8fce\u56de\u5230\u6211\u4eec\u7684\u201c\u4f18\u5316\u5de5\u4f5c\u6d41\u201d\u7cfb\u5217\uff01 \u6211\u4eec\u6536\u5230\u4e86\u4e00\u4e9b\u6765\u81ea\u793e\u533a\u7684\u8bf7\u6c42\uff0c\u7528\u6237\u4eec\u5e0c\u671b\u8ba8\u8bba\u53ef\u7531 QA \u56e2\u961f\u4f7f\u7528\u7684 YouTrack \u529f\u80fd\u3002 \u73b0\u5728\uff0c\u6211\u4eec\u5c06\u4ecb\u7ecd\u5982\u4f55\u5728 YouTrack \u4e2d\u7ef4\u62a4\u6d4b\u8bd5\u7ba1\u7406\u65b9\u6848 (TMS)\u3002 \u672c\u6587\u5c06\u6f14\u793a\u5982\u4f55\u8bbe\u7f6e YouTrack \u4ee5\u4f7f\u60a8\u80fd\u591f\u5728 YouTrack \u5185\u90e8\u7ef4\u62a4\u6d4b\u8bd5\u7ba1\u7406\u529f\u80fd\uff0c\u4ece\u800c\u907f\u514d\u4f7f\u7528\u7b2c\u4e09\u65b9\u89e3\u51b3\u65b9\u6848\u3002 \u8fd9\u79cd\u65b9\u5f0f\u53ef\u5e2e\u52a9\u60a8\u8282\u7701\u8bb8\u53ef\u8bc1\u6210\u672c\uff0c\u5e76\u4f7f\u6d4b\u8bd5\u7ba1\u7406\u6d41\u7a0b\u66f4\u52a0\u987a\u7545\u3002<\/p>\n<p>\u672c\u6587\u9002\u7528\u4e8e\u6d4b\u8bd5\u7ecf\u7406\u4ee5\u53ca\u4efb\u4f55\u5bf9 YouTrack \u5de5\u4f5c\u6d41\u611f\u5174\u8da3\u7684\u4eba\u5458\u3002 \u6587\u7ae0\u4e3a\u60a8\u63d0\u4f9b\u4e86\u4e00\u4e2a\u6709\u5173\u5982\u4f55\u5728 YouTrack \u4e2d\u6784\u5efa\u6d4b\u8bd5\u6d41\u7a0b\u7684\u5b9e\u7528\u793a\u4f8b\uff0c\u5176\u4e2d\u5305\u542b\u9488\u5bf9\u9879\u76ee\u548c\u95ee\u9898\u8bbe\u7f6e\u7684\u5efa\u8bae\uff0c\u4ee5\u53ca\u6709\u5173\u5982\u4f55\u5b9e\u73b0\u6d4b\u8bd5\u7ba1\u7406\u65b9\u6848\u7684\u8bf4\u660e\u7b49\u5185\u5bb9\u3002<\/p>\n<p>\u6211\u4eec\u8fd8\u7cbe\u5fc3\u7f16\u6392\u4e86\u4e00\u7cfb\u5217\u53ef\u7acb\u5373\u4f7f\u7528\u7684\u5de5\u4f5c\u6d41\u4ee3\u7801\u5757\uff0c\u53ef\u7528\u4e8e\u81ea\u52a8\u5316\u6d4b\u8bd5\u7ba1\u7406\u6d41\u7a0b\u3002 \u8fd9\u4e9b\u4ee3\u7801\u5757\u53ef\u6709\u6548\u7b80\u5316\u5c06\u6d4b\u8bd5\u7528\u4f8b\u4e0e\u7279\u5b9a\u6d4b\u8bd5\u8fd0\u884c\u76f8\u5173\u8054\u3001\u514b\u9686\u6d4b\u8bd5\u8fd0\u884c\u3001\u663e\u793a\u6709\u5173\u4e0b\u6b21\u6d4b\u8bd5\u7684\u7cfb\u7edf\u5efa\u8bae\u7b49\u6d41\u7a0b\u3002<br \/>\n<a name=\"ref1\"><\/a><\/p>\n<h2>YouTrack \u4e2d\u9488\u5bf9\u6d4b\u8bd5\u7ba1\u7406\u9879\u76ee\u7684\u8bbe\u7f6e<\/h2>\n<p>\u60a8\u53ef\u80fd\u9700\u8981\u4e3a\u6d4b\u8bd5\u7ba1\u7406\u9879\u76ee\u4f7f\u7528\u7279\u5b9a\u4e8e\u6d4b\u8bd5\u7ba1\u7406\u7684\u95ee\u9898\u7c7b\u578b\uff0c\u4f8b\u5982\u6d4b\u8bd5\u7528\u4f8b\u3001\u6d4b\u8bd5\u5957\u4ef6\uff08\u4e00\u7ec4\u6d4b\u8bd5\u7528\u4f8b\uff09\u3001\u6d4b\u8bd5\u8fd0\u884c\uff08\u4e00\u7ec4\u4e3a\u7279\u5b9a\u6d4b\u8bd5\u5468\u671f\u5206\u914d\u7684\u6d4b\u8bd5\u7528\u4f8b\u6216\u6d4b\u8bd5\u5957\u4ef6\uff09\u4ee5\u53ca\u5206\u914d\u7ed9\u7279\u5b9a\u6d4b\u8bd5\u8fd0\u884c\u7684\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u3002 \u60a8\u5e94<a href=\"https:\/\/www.jetbrains.com\/help\/youtrack\/standalone\/Manage-Custom-Fields-Per-Project.html#add-custom-field-to-project\" target=\"_blank\" rel=\"noopener\">\u914d\u7f6e<\/a> TMS \u9879\u76ee\u6240\u9700\u7684\u95ee\u9898\u7c7b\u578b\u3002 \u4ee5\u4e0b 3 \u79cd<a href=\"https:\/\/www.jetbrains.com\/help\/youtrack\/standalone\/Link-Types.html\" target=\"_blank\" rel=\"noopener\">\u95ee\u9898\u94fe\u63a5<\/a>\u7c7b\u578b\u53ef\u7528\u4e8e\u5728\u60a8\u7684\u6d4b\u8bd5\u7ba1\u7406\u4efb\u52a1\u4e4b\u95f4\u5efa\u7acb\u8fde\u63a5\u548c\u76f8\u5173\u6027\uff1a<\/p>\n<ul>\n<li>\u6807\u51c6\u201cparent-subtasks\u201d\uff0c\u53ef\u7528\u4e8e\u7ef4\u62a4\u6d4b\u8bd5\u8fd0\u884c\u4e0e\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u4ee5\u53ca\u6d4b\u8bd5\u5957\u4ef6\u4e0e\u6d4b\u8bd5\u7528\u4f8b\u4e4b\u95f4\u7684\u5173\u7cfb\u3002<\/li>\n<li>\u81ea\u5b9a\u4e49\u201cExecution\u201d\uff0c\u53ef\u7528\u4e8e\u7ef4\u62a4\u6d4b\u8bd5\u7528\u4f8b\u4e0e\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u4e4b\u95f4\u7684\u5173\u7cfb\u3002<\/li>\n<li>\u81ea\u5b9a\u4e49\u201cRelated Bug\u201d\uff0c\u53ef\u7528\u4e8e\u7ef4\u62a4\u5931\u8d25\u7684\u6d4b\u8bd5\u4e0e\u5206\u914d\u7684\u9519\u8bef\u4e4b\u95f4\u7684\u5173\u7cfb\u3002<\/li>\n<\/ul>\n<p>\u6d4b\u8bd5\u8981\u6c42\u53ef\u4f5c\u4e3a\u5728\u6587\u672c\u5b57\u6bb5\u4e2d\u94fe\u63a5\u5230\u95ee\u9898\u7684<a href=\"https:\/\/www.jetbrains.com\/help\/youtrack\/standalone\/create-articles.html\" target=\"_blank\" rel=\"noopener\">\u6587\u7ae0<\/a>\u8fdb\u884c\u7ef4\u62a4\u3002<br \/>\n\u6b64\u5916\uff0c\u5bf9\u4e8e\u60a8\u7684\u6d4b\u8bd5\u7ba1\u7406\u4efb\u52a1\uff0c\u60a8\u53ef\u80fd\u9700\u8981\u8bbe\u7f6e\u81ea\u5b9a\u4e49\u5b57\u6bb5\uff08\u4f8b\u5982 Test mode\u3001Category\u3001Test flow \u548c Application\uff09\uff0c\u4ee5\u53ca\u9884\u5b9a\u4e49\u503c\uff08\u4f8b\u5982\u9884\u5b9a\u4e49\u6d4b\u8bd5\u72b6\u6001\uff09\u3002 \u9700\u8981\u4e13\u95e8\u663e\u793a\u4e00\u79cd\u95ee\u9898\u7c7b\u578b\u7684\u4fe1\u606f\u65f6\uff0c\u60a8\u4e5f\u53ef\u4ee5\u4f7f\u7528\u6761\u4ef6\u5b57\u6bb5\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS0_1.png\" alt=\"\" \/><\/p>\n<p>\u60a8\u53ef\u4ee5\u6839\u636e\u60a8\u7684\u4e1a\u52a1\u9700\u6c42\u8c03\u6574\u201cTest Case\u201d\u548c\u201cTest Run\u201d\u95ee\u9898\u7c7b\u578b\u7684\u4e00\u7ec4\u9884\u5b9a\u4e49\u5b57\u6bb5\u3002<\/p>\n<h2>\u8bbe\u7f6e\u6d4b\u8bd5\u8fd0\u884c\u548c\u6d4b\u8bd5\u6267\u884c<\/h2>\n<p>\u4e3e\u4f8b\u6765\u8bf4\uff0c\u5f53\u6211\u4eec\u9700\u8981\u6d4b\u8bd5\u7279\u5b9a\u4ea7\u54c1\u7248\u672c\u65f6\uff0c\u6211\u4eec\u9700\u8981\u521b\u5efa\u4e00\u4e2a\u6d4b\u8bd5\u8fd0\u884c\uff0c\u5e76\u4e3a\u5176\u5206\u914d\u4e00\u5b9a\u6570\u91cf\u7684\u76f8\u5173\u6d4b\u8bd5\u5957\u4ef6\u548c\u6d4b\u8bd5\u7528\u4f8b\u3002 \u6240\u9700\u6b65\u9aa4\u5982\u4e0b\uff1a<\/p>\n<ul>\n<li>\u521b\u5efa\u95ee\u9898\u7c7b\u578b\u4e3a\u201cTest Run\u201d\u7684\u95ee\u9898\u3002<\/li>\n<li>\u4f7f\u7528 \u201cAssigned test case and test suite&#8221; \u81ea\u5b9a\u4e49\u94fe\u63a5\u7c7b\u578b\u4ee5\u5173\u8054\u5230\u95ee\u9898\u3002<\/li>\n<li>\u5728\u5f39\u51fa\u7a97\u53e3\u4e2d\u6307\u5b9a\u8981\u5728\u6d4b\u8bd5\u8fd0\u884c\u4e2d\u5305\u542b\u7684\u6d4b\u8bd5\u7528\u4f8b\u3002<\/li>\n<\/ul>\n<p>\u201cPopulate Test Run\u201d \u5de5\u4f5c\u6d41\u7684\u6b65\u9aa4\u5c06\u4e3a\u60a8\u7684 QA \u56e2\u961f\u8282\u7701\u65f6\u95f4\u548c\u7cbe\u529b\uff1a<\/p>\n<ul>\n<li>\u5c06\u5728\u7cfb\u7edf\u4e2d\u521b\u5efa\u6240\u6709\u9009\u5b9a\u6d4b\u8bd5\u5957\u4ef6\u548c\u6d4b\u8bd5\u7528\u4f8b\u7684\u526f\u672c\u3002 \u6240\u6709\u95ee\u9898\u90fd\u5c06\u5177\u6709\u201cTest Case Execution\u201d\u95ee\u9898\u7c7b\u578b\u3002 <\/li>\n<li>\u201cExecution\u201d\u94fe\u63a5\u7c7b\u578b\u4f1a\u5c06\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u8fde\u63a5\u81f3\u9009\u5b9a\u7684\u6d4b\u8bd5\u7528\u4f8b\u3002<\/li>\n<li>\u65b0\u521b\u5efa\u7684\u95ee\u9898\u5c06\u94fe\u63a5\u81f3\u4f5c\u4e3a\u7236\u95ee\u9898\u7684\u6d4b\u8bd5\u8fd0\u884c\u53ca\u5176\u76f8\u5173\u5b50\u4efb\u52a1\u3002<\/li>\n<\/ul>\n<p>\u5de5\u4f5c\u65b9\u5f0f\u5982\u4e0b\uff1a<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done1s720cut.gif\" alt=\"\" \/><\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\nvar entities = require(&#039;@jetbrains\/youtrack-scripting-api\/entities&#039;);\nvar workflow = require(&#039;@jetbrains\/youtrack-scripting-api\/workflow&#039;);\n\nexports.rule = entities.Issue.onChange({\n  title: &#039;Populate-test-run&#039;,\n  guard: function(ctx) {\n    var issue = ctx.issue;\n    return !issue.isChanged(&#039;project&#039;) &amp;&amp; issue.Type &amp;&amp; (issue.Type.name == ctx.Type.TestRun.name) &amp;&amp; issue.links[ctx.Execution.outward].added.isNotEmpty() &amp;&amp; issue.isReported;\n  },\n  action: function(ctx) {\n    var issue = ctx.issue;\n    var totalTestRuns = issue.links[ctx.Execution.outward].added.size;\n    issue.links[ctx.Execution.outward].added.forEach(function(TestCase) {\n      TestCase.links[ctx.Execution.inward].delete(issue);\n\n      var TestCaseRun = TestCase.copy();\n      TestCaseRun.Type = ctx.Type.TestExecution.name;\n      TestCaseRun.Status = ctx.Status.NoRun.name;\n      Object.keys(TestCaseRun.links).forEach(function(linkType) {\n       if (!TestCaseRun.links[linkType])\n        return;\n         TestCaseRun.links[linkType].clear();\n      });\n      TestCaseRun.summary = &quot;[TEST_CASE_EXECUTION&quot; + &quot;] [&quot; + TestCaseRun.summary + &quot;]&quot;;\n\n      TestCaseRun.links[ctx.Subtask.inward].add(issue);\n      issue.links[ctx.Subtask.outward].add(TestCaseRun);\n      TestCaseRun.links[ctx.Execution.outward].add(TestCase);\n    });\n    issue.fields[&#039;Total number of test cases&#039;] = totalTestRuns;\n  },\n  requirements: {\n    Execution: {\n      type: entities.IssueLinkPrototype,\n      name: &#039;Execution&#039;,\n      inward: &#039;Execution&#039;,\n      outward: &#039;Assigned test case or test suite&#039;\n    },\n    Subtask: {\n      type: entities.IssueLinkPrototype,\n      name: &#039;Subtask&#039;,\n      inward: &#039;parent for&#039;,\n      outward: &#039;subtask of&#039;\n    },\n    Type: {\n      type: entities.EnumField.fieldType,\n      TestExecution: {\n        name: &quot;Test Case Execution&quot;\n      },\n      TestRun: {\n        name: &quot;Test Run&quot;\n      },\n      TestCase: {\n        name: &quot;Test Case&quot;\n      },\n      TestSuite: {\n        name: &quot;Test Suite&quot;\n      }\n    },\n    Total: {\n      type: entities.Field.integerType,\n      name: &#039;Total number of test cases&#039;\n    },\n    TotalFailed: {\n      type: entities.Field.integerType,\n      name: &#039;Number of failed test cases&#039;\n    },\n    TotalPassed: {\n      type: entities.Field.integerType,\n      name: &#039;Number of passed test cases&#039;\n    },\n    Status: {\n      type: entities.EnumField.fieldType,\n      InProgress: {\n        name: &#039;In Progress&#039;\n      },\n      Passed: {\n        name: &#039;Passed&#039;\n      },\n      Failed: {\n        name: &#039;Failed&#039;\n      },\n      NoRun: {\n        name: &#039;No Run&#039;\n      },\n    },\n  }\n});\n<\/pre>\n<\/details>\n<h2>\u7ef4\u62a4\u6d4b\u8bd5\u6d3b\u52a8<\/h2>\n<p>\u6d4b\u8bd5\u8fd0\u884c\u8bbe\u7f6e\u5168\u90e8\u5b8c\u6210\u540e\uff0c\u6d4b\u8bd5\u5de5\u7a0b\u5e08\u5373\u53ef\u5f00\u59cb\u6d4b\u8bd5\u3002 \u9ed8\u8ba4\u60c5\u51b5\u4e0b\uff0c\u6240\u6709\u201cTest Case Execution\u201d\u548c\u201cTest Run\u201d\u7c7b\u578b\u7684\u95ee\u9898\u5747\u5904\u4e8e\u201cNo Run\u201d\u72b6\u6001\u3002<\/p>\n<h3>\u5982\u4f55\u5728\u6d4b\u8bd5\u4e4b\u95f4\u8fdb\u884c\u5207\u6362<\/h3>\n<p>\u4f7f\u7528\u4e00\u7ec4\u6d4b\u8bd5\u65f6\uff0c\u60a8\u53ef\u4ee5\u901a\u8fc7\u4e24\u79cd\u65b9\u5f0f\u4ece\u4e00\u9879\u6d4b\u8bd5\u5207\u6362\u5230\u53e6\u4e00\u9879\u6d4b\u8bd5\uff1a<\/p>\n<ol>\n<li>\u5728\u95ee\u9898\u5217\u8868\u9875\u9762\u4e0a\u624b\u52a8\u66f4\u6539\u6d4b\u8bd5\u72b6\u6001\u3002<\/li>\n<li>\u6253\u5f00\u4e00\u9879\u6d4b\u8bd5\uff0c\u7136\u540e\u5207\u6362\u5230\u4e0b\u4e00\u9879\u672a\u5b8c\u6210\u7684\u6d4b\u8bd5\u3002 \u5f53\u60a8\u5e0c\u671b\u4ee5\u8fd9\u79cd\u65b9\u5f0f\u5207\u6362\u6d4b\u8bd5\u65f6\uff0c\u5c06\u51fa\u73b0\u4e00\u4e2a\u6709\u7528\u7684\u5f39\u51fa\u7a97\u53e3\uff0c\u5efa\u8bae\u8fd0\u884c\u4e0b\u4e00\u4e2a\u6d4b\u8bd5\u3002<br \/>\n    \u53ef\u4ee5\u81ea\u52a8\u6267\u884c\u4ee5\u4e0b\u64cd\u4f5c\uff1a<\/p>\n<ul>\n<li> \u68c0\u67e5\u662f\u5426\u5b58\u5728\u5904\u4e8e\u201cNo Run\u201d\u72b6\u6001\u4e14\u5c5e\u4e8e\u540c\u4e00\u6d4b\u8bd5\u8fd0\u884c\u7684\u6d4b\u8bd5\u3002<\/li>\n<li>\u663e\u793a\u4e00\u6761\u6d88\u606f\uff0c\u5176\u4e2d\u5305\u542b\u4e0b\u4e00\u9879\u53ef\u7528\u6d4b\u8bd5\uff08\u5982\u6709\uff09\u7684\u7f51\u5740\u3002\n<p>    <a href=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done3s720.gif\" rel=\"attachment wp-att-6366 noopener\" target=\"_blank\"><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done3s720.gif\" alt=\"ApplyingCommandForSubsystem\"  \/><\/a><\/li>\n<\/ul>\n<p> \u53ef\u4f7f\u7528\u4ee5\u4e0b\u4ee3\u7801\u5b9e\u73b0\u4e0a\u8ff0\u64cd\u4f5c\u3002<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\n action: function(ctx) {\n  var issue = ctx.issue;\n  if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var resultSet = null;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if (v.Status.name == ctx.Status.Failed.name) {\n                  isPassing = false;\n                } else if ((v.Status.name == ctx.Status.InProgress.name) &amp;&amp; (v.id !== issue.id)) {\n                  resultSet = v;\n                }\n              });\n              if (resultSet) {\n                var otherIssueLink = &#039;&lt;a href=&quot;&#039; + resultSet.url + &#039;&quot;&gt; &#039; + resultSet.id + &#039;&lt;\/a&gt;&#039;;\n                var message = &#039;Switch to next open test in current Test Run&#039; + otherIssueLink + &#039;.&#039;;\n                workflow.message(message);\n    }\n}\n}\n<\/pre>\n<\/details>\n<\/ol>\n<h2>\u6d4b\u8bd5\u72b6\u6001 <\/h2>\n<p><a name=\"ref2\"><\/a><\/p>\n<p>\u4ece\u4e00\u9879\u6d4b\u8bd5\u5207\u6362\u5230\u53e6\u4e00\u9879\u6d4b\u8bd5\u65f6\uff0c\u6d4b\u8bd5\u5de5\u7a0b\u5e08\u53ef\u4ee5\u6307\u793a\u6d4b\u8bd5\u201cPassed\u201d\uff0c\u4e5f\u53ef\u4ee5\u5c06\u6d4b\u8bd5\u72b6\u6001\u66f4\u6539\u4e3a\u201cFailed\u201d\uff08\u5982\u679c\u5728\u6267\u884c\u6d4b\u8bd5\u7528\u4f8b\u671f\u95f4\u8bc6\u522b\u5230\u9519\u8bef\uff09\u3002 \u57fa\u4e8e\u6211\u4eec\u6700\u521d\u4e3a\u6b64\u9879\u76ee\u914d\u7f6e\u7684<a href=\"#ref1\"> YouTrack \u8bbe\u7f6e<\/a>\uff0c\u6709\u591a\u79cd\u9884\u5b9a\u4e49\u6d4b\u8bd5\u8fd0\u884c\u72b6\u6001\uff1a<\/p>\n<ul>\n<li>Failing\uff1a\u81f3\u5c11\u6709\u4e00\u9879\u5173\u8054\u6d4b\u8bd5\u7684\u72b6\u6001\u4e3a\u201cNo Run\u201d\uff0c\u5e76\u4e14\u81f3\u5c11\u6709\u4e00\u9879\u5173\u8054\u6d4b\u8bd5\u7684\u72b6\u6001\u4e3a\u201cFailed\u201d\u3002<\/li>\n<li>Passing\uff1a\u81f3\u5c11\u6709\u4e00\u9879\u5173\u8054\u6d4b\u8bd5\u7684\u72b6\u6001\u4e3a\u201cNo Run\u201d\uff0c\u5e76\u4e14\u6ca1\u6709\u72b6\u6001\u4e3a\u201cFailed\u201d\u7684\u5173\u8054\u6d4b\u8bd5\u3002<\/li>\n<li>Failed\uff1a\u6ca1\u6709\u72b6\u6001\u4e3a\u201cNo Run\u201d\u7684\u5173\u8054\u6d4b\u8bd5\uff0c\u5e76\u4e14\u81f3\u5c11\u6709\u4e00\u9879\u5173\u8054\u6d4b\u8bd5\u7684\u72b6\u6001\u4e3a\u201cFailed\u201d\u3002<\/li>\n<li>Passed\uff1a\u6ca1\u6709\u72b6\u6001\u4e3a\u201cNo Run\u201d\u6216\u201cFailed\u201d\u7684\u5173\u8054\u6d4b\u8bd5\u3002<\/li>\n<\/ul>\n<p><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done2s720.gif\" alt=\"\" \/><\/p>\n<p>\u4e0a\u9762\u5217\u51fa\u7684\u72b6\u6001\u53ef\u4ee5\u5e2e\u52a9\u60a8\u7684 QA \u56e2\u961f\u5728\u6574\u4e2a\u6d4b\u8bd5\u5468\u671f\u5185\u76d1\u63a7\u6d4b\u8bd5\u8fdb\u5ea6\u3002 \u9700\u8981\u6ce8\u610f\u7684\u662f\uff0c\u6d4b\u8bd5\u8fd0\u884c\u72b6\u6001\u4ec5\u53d6\u51b3\u4e8e\u5173\u8054\u6d4b\u8bd5\uff0c\u4e0d\u5e94\u624b\u52a8\u66f4\u6539\u3002 \u9700\u8981\u4f7f\u7528\u6309<a href=\"href=\"https:\/\/www.jetbrains.com\/help\/youtrack\/standalone\/state-machine-per-issue-type.html#state-machine-per-issue-type2\"\">\u95ee\u9898\u7c7b\u578b\u7684\u72b6\u6001\u673a<\/a>\u5de5\u4f5c\u6d41\u6765\u5b9e\u73b0\u786e\u5b9a\u6d4b\u8bd5\u8fd0\u884c\u72b6\u6001\u7684\u8fc7\u7a0b\u3002 \u5de5\u4f5c\u6d41\u4e2d\u4e5f\u5305\u542b\u4e86\u6d4b\u8bd5\u5f00\u5173\u4ee3\u7801\u5757\u3002<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\n\nvar entities = require(&#039;@jetbrains\/youtrack-scripting-api\/entities&#039;);\nvar workflow = require(&#039;@jetbrains\/youtrack-scripting-api\/workflow&#039;);\n\nexports.rule = entities.Issue.stateMachine({\n  title: &#039;status-management&#039;,\n  stateFieldName: &#039;Status&#039;,\n  typeFieldName: &#039;Type&#039;,\n  defaultMachine: {\n    &#039;No Run&#039;: {\n      initial: true,\n      transitions: {\n        &#039;Failed&#039;: {\n          targetState: &#039;Failed&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var resultSet = null;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if (v.Status.name == ctx.Status.Failed.name) {\n                  isPassing = false;\n                } else if ((v.Status.name == ctx.Status.InProgress.name) &amp;&amp; (v.id !== issue.id)) {\n                  resultSet = v;\n                }\n              });\n              if (resultSet) {\n                var otherIssueLink = &#039;&lt;a href=&quot;&#039; + resultSet.url + &#039;&quot;&gt; &#039; + resultSet.id + &#039;&lt;\/a&gt;&#039;;\n                var message = &#039;Switch to next open test in current Test Run&#039; + otherIssueLink + &#039;.&#039;;\n                workflow.message(message);\n                \/\/ Updating Test Run Status \n                parent.fields[&quot;Status&quot;] = ctx.Status.Failing;\n              } else {\n                parent.fields[&quot;Status&quot;] = ctx.Status.Failed;\n              }\n            }\n          }\n        },\n        &#039;Passed&#039;: {\n          guard: function(ctx) {\n            var issue = ctx.issue;\n            return !issue.isChanged(&#039;project&#039;) &amp;&amp; !issue.becomesReported &amp;&amp; issue.isReported &amp;&amp; (issue.Type.name == ctx.Type.TestExecution.name);\n          },\n          targetState: &#039;Passed&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var resultSet = null;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if (v.Status.name == ctx.Status.Failed.name) {\n                  isPassing = false;\n                } else if ((v.Status.name == ctx.Status.InProgress.name) &amp;&amp; (v.id !== issue.id)) {\n                  resultSet = v;\n                }\n              });\n              if (resultSet) {\n                var otherIssueLink = &#039;&lt;a href=&quot;&#039; + resultSet.url + &#039;&quot;&gt; &#039; + resultSet.id + &#039;&lt;\/a&gt;&#039;;\n                var message = &#039;Switch to next open test in current Test Run&#039; + otherIssueLink + &#039;.&#039;;\n                workflow.message(message);\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passing : ctx.Status.Failing;\n              } else {\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passed : ctx.Status.Failed;\n              }\n            }\n          }\n        }\n      }\n    },\n    Passed: {\n      transitions: {\n        &#039;Failed&#039;: {\n          guard: function(ctx) {\n            var issue = ctx.issue;\n            return !issue.isChanged(&#039;project&#039;) &amp;&amp; !issue.becomesReported &amp;&amp; issue.isReported &amp;&amp; (issue.Type.name == ctx.Type.TestExecution.name);\n          },\n          targetState: &#039;Failed&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n             var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var resultSet = null;\n              TestRunList.forEach(function(v) {\n                if (v.Status.name == ctx.Status.Failed.name) {\n                } else if ((v.Status.name == ctx.Status.InProgress.name) &amp;&amp; (v.id !== issue.id)) {\n                  resultSet = v;\n                }\n              });\n              if (resultSet) {\n                var otherIssueLink = &#039;&lt;a href=&quot;&#039; + resultSet.url + &#039;&quot;&gt; &#039; + resultSet.id + &#039;&lt;\/a&gt;&#039;;\n                var message = &#039;Switch to next open test in current Test Run&#039; + otherIssueLink + &#039;.&#039;;\n                workflow.message(message);\n                parent.fields[&quot;Status&quot;] = ctx.Status.Failing;\n              } else {\n                parent.Status = ctx.Status.Failed;\n              }\n            }\n          }\n        },\n        &#039;No Run&#039;: {\n          guard: function(ctx) {\n            var issue = ctx.issue;\n            return !issue.isChanged(&#039;project&#039;) &amp;&amp; !issue.becomesReported &amp;&amp; issue.isReported &amp;&amp; (issue.Type.name == ctx.Type.TestExecution.name);\n          },\n          targetState: &#039;No Run&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var ActiveTestRun = false;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if (v.Status.name == ctx.Status.Failed.name) {\n                  isPassing = false;\n                  ActiveTestRun = true;\n                } else if ((v.Status.name == ctx.Status.Passed.name) &amp;&amp; (v.id !== issue.id)) {\n                  ActiveTestRun = true;\n                }\n              });\n              if (ActiveTestRun) {\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passing : ctx.Status.Failing;\n              } else parent.fields[&quot;Status&quot;] = ctx.Status.InProgress;\n            }\n          }\n        }\n      }\n    },\n    Failed: {\n      transitions: {\n        &#039;Passed&#039;: {\n          guard: function(ctx) {\n            var issue = ctx.issue;\n            return !issue.isChanged(&#039;project&#039;) &amp;&amp; !issue.becomesReported &amp;&amp; issue.isReported &amp;&amp; (issue.Type.name == ctx.Type.TestExecution.name);\n          },\n          targetState: &#039;Passed&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var resultSet = null;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if ((v.Status.name == ctx.Status.Failed.name) &amp;&amp; (v.id !== issue.id)) {\n                  isPassing = false;\n                } else if ((v.Status.name == ctx.Status.InProgress.name) &amp;&amp; (v.id !== issue.id)) {\n                  resultSet = v;\n                }\n              });\n              if (resultSet) {\n                var otherIssueLink = &#039;&lt;a href=&quot;&#039; + resultSet.url + &#039;&quot;&gt; &#039; + resultSet.id + &#039;&lt;\/a&gt;&#039;;\n                var message = &#039;Switch to next open test in current Test Run&#039; + otherIssueLink + &#039;.&#039;;\n                workflow.message(message);\n\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passing : ctx.Status.Failing;\n              } else {\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passed : ctx.Status.Failed;\n              }\n            }\n          }\n        },\n        &#039;No Run&#039;: {\n          guard: function(ctx) {\n            var issue = ctx.issue;\n            return !issue.isChanged(&#039;project&#039;) &amp;&amp; !issue.becomesReported &amp;&amp; issue.isReported &amp;&amp; (issue.Type.name == ctx.Type.TestExecution.name);\n          },\n          targetState: &#039;No Run&#039;,\n          action: function(ctx) {\n            var issue = ctx.issue;\n            if (!issue.links[&#039;subtask of&#039;].isEmpty) {\n              var parent = issue.links[&#039;subtask of&#039;].first();\n              var TestRunList = parent.links[ctx.Subtask.outward];\n              var ActiveTestRun = false;\n              var isPassing = true;\n              TestRunList.forEach(function(v) {\n                if ((v.Status.name == ctx.Status.Failed.name) &amp;&amp; (v.id !== issue.id)) {\n                  isPassing = false;\n                  ActiveTestRun = true;\n                } else if ((v.Status.name == ctx.Status.Passed.name) &amp;&amp; (v.id !== issue.id)) {\n                  ActiveTestRun = true;\n                }\n              });\n              if (ActiveTestRun) {\n                parent.fields[&quot;Status&quot;] = (isPassing) ? ctx.Status.Passing : ctx.Status.Failing;\n              } else parent.fields[&quot;Status&quot;] = ctx.Status.InProgress;\n            }\n          }\n        }\n      }\n    }\n  },\n  alternativeMachines: {\n    &#039;Test Run&#039;: {\n      &#039;No Run&#039;: {\n        initial: true,\n        transitions: {\n          &#039;Failing&#039;: {\n            targetState: &#039;Failing&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          },\n          &#039;Failed&#039;: {\n            targetState: &#039;Failed&#039;,\n            action: function(ctx) {\n         \/* Add actions. *\/\n            }\n          },\n          &#039;Passing&#039;: {\n            targetState: &#039;Passing&#039;,\n            action: function(ctx) {\n          \/* Add actions. *\/\n            }\n          },\n          &#039;Passed&#039;: {\n            targetState: &#039;Passed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          }\n        }\n      },\n      Failing: {\n        transitions: {\n          &#039;Passing&#039;: {\n            targetState: &#039;Passing&#039;,\n            action: function(ctx) {\n           \/* Add actions . *\/\n            }\n          },\n          &#039;Passed&#039;: {\n            targetState: &#039;Passed&#039;,\n            action: function(ctx) {\n          \/* Add actions. *\/\n            }\n          },\n          &#039;Failed&#039;: {\n            targetState: &#039;Failed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          }\n        }\n      },\n      Passing: {\n        transitions: {\n          &#039;Failing&#039;: {\n            targetState: &#039;Passing&#039;,\n            action: function(ctx) {\n              workflow.check(false, workflow.i18n(&#039;Test Run has-read-only status which is defined based on assigned tests statuses&#039;));\n            }\n          },\n          &#039;Passed&#039;: {\n            targetState: &#039;Passed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          },\n          &#039;Failed&#039;: {\n            targetState: &#039;Failed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          }\n        }\n      },\n      Failed: {\n        transitions: {\n          &#039;Passing&#039;: {\n            targetState: &#039;Passing&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          },\n          &#039;Passed&#039;: {\n            targetState: &#039;Passed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          },\n          &#039;Failing&#039;: {\n            targetState: &#039;Failed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          }\n        }\n      },\n      Passed: {\n        transitions: {\n          &#039;Passing&#039;: {\n            targetState: &#039;Passing&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          },\n          &#039;Failed&#039;: {\n            targetState: &#039;Passed&#039;,\n            action: function(ctx) {\n          \/* Add actions. *\/\n            },\n          },\n          &#039;Failing&#039;: {\n            targetState: &#039;Failed&#039;,\n            action: function(ctx) {\n           \/* Add actions. *\/\n            }\n          }\n        }\n      }\n    }\n  },\n  requirements: {\n    Assignee: {\n      type: entities.User.fieldType\n    },\n    Status: {\n      type: entities.EnumField.fieldType,\n      InProgress: {\n        name: &#039;No Run&#039;\n      },\n      Failing: {\n        name: &#039;Failing&#039;\n      },\n      Passing: {\n        name: &#039;Passing&#039;\n      },\n      Passed: {\n        name: &#039;Passed&#039;\n      },\n      Failed: {\n        name: &#039;Failed&#039;\n      },\n    },\n    Type: {\n      type: entities.EnumField.fieldType,\n      TestRun: {\n        name: &quot;Test Run&quot;\n      },\n      TestExecution: {\n        name: &quot;Test Case Execution&quot;\n      }\n    },\n    Subtask: {\n      type: entities.IssueLinkPrototype,\n      name: &#039;Subtask&#039;,\n      inward: &#039;subtask of&#039;,\n      outward: &#039;parent for&#039;\n    },\n  }\n});\n <\/pre>\n<\/details>\n<h1>\u6d4b\u8bd5\u7edf\u8ba1\u4fe1\u606f<\/h1>\n<p>\u60a8\u53ef\u80fd\u4e5f\u6709\u5174\u8da3\u4e86\u89e3\u5173\u952e\u6d4b\u8bd5\u5468\u671f\u7edf\u8ba1\u4fe1\u606f\u3002 \u51fa\u4e8e\u6f14\u793a\u76ee\u7684\uff0c\u6211\u4eec\u5305\u542b\u4e86\u4ee5\u4e0b\u6307\u6807\uff1a<\/p>\n<ul>\n<li>\u5206\u914d\u7ed9\u7279\u5b9a\u6d4b\u8bd5\u8fd0\u884c\u7684\u6d4b\u8bd5\u603b\u6570\u3002<\/li>\n<li>\u72b6\u6001\u4e3a\u201cPassed\u201d\u7684\u6d4b\u8bd5\u7684\u6570\u91cf\u3002<\/li>\n<li>\u72b6\u6001\u4e3a\u201cFailed\u201d\u7684\u6d4b\u8bd5\u7684\u6570\u91cf\u3002<\/li>\n<\/ul>\n<p><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done5s720.gif\" alt=\"\" \/><br \/>\n\u57fa\u4e8e\u4e0b\u65b9\u4ee3\u7801\uff0c\u5728\u53d1\u751f\u4ee5\u4e0b\u4efb\u4f55\u53d8\u66f4\u65f6\uff0c\u6240\u6709\u6307\u6807\u90fd\u5c06\u66f4\u65b0\uff1a\u6d4b\u8bd5\u72b6\u6001\u53d8\u66f4\u3001\u5bf9\u6d4b\u8bd5\u8fd0\u884c\u5206\u914d\u65b0\u6d4b\u8bd5\u4ee5\u53ca\u4ece\u6d4b\u8bd5\u8fd0\u884c\u79fb\u9664\u6d4b\u8bd5\u3002<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\n  exports.calculateStatuses = function(parent) {\n  var totalTR = 0;\n  var totalFailed = 0;\n  var totalPassed = 0;\n  if (!parent.links[&#039;parent for&#039;]) {\n    return;\n  } else {\n    parent.links[&#039;parent for&#039;].forEach(function(tr) {\n      totalTR++;\n      if (tr.Status.name == &#039;Passed&#039;) {\n        totalFailed++;\n      }\n      if (tr.Status.name == &#039;Failed&#039;) {\n        totalPassed++;\n      }\n    });\n    parent.fields[&#039;Total number of test cases&#039;] = totalTR;\n    parent.fields[&#039;Number of passed test cases&#039;] = totalPassed;\n    parent.fields[&#039;Number of failed test cases&#039;] = totalFailed;\n    return true;\n  }\n};\nexports.resetStatuses = function(testRun, testRunCopy) {\n   testRunCopy.fields[&#039;Total number of test cases&#039;] = testRun.fields[&#039;Total number of test cases&#039;];\n   testRunCopy.fields[&#039;Number of passed test cases&#039;] = 0;\n   testRunCopy.fields[&#039;Number of failed test cases&#039;] = 0;\n  return true;\n};\n<\/pre>\n<\/details>\n<p>\u5728\u6211\u4eec\u7684\u7528\u4f8b\u4e2d\uff0c\u6b64\u4ee3\u7801\u5728\u591a\u4e2a\u5de5\u4f5c\u6d41\u89c4\u5219\u4e0b\u4f1a\u88ab\u89e6\u53d1\uff0c\u4f8b\u5982\u201cUpdate stats when links are adjusted\u201d\u3001\u201cSwitch to the next test case\u201d\u7b49\u3002 \u5982\u679c\u8981\u91c7\u7528\u4e00\u7ec4\u80fd\u591f\u6ee1\u8db3\u60a8\u7279\u5b9a\u9700\u6c42\u7684\u6307\u6807\uff0c\u60a8\u53ef\u4ee5\u6dfb\u52a0\u6240\u9700\u7684\u81ea\u5b9a\u4e49\u5b57\u6bb5\u5e76\u8c03\u6574\u60a8\u7684\u5de5\u4f5c\u6d41\u903b\u8f91\u3002<\/p>\n<h2>\u95ee\u9898\u8ddf\u8e2a\u5668\u529f\u80fd<\/h2>\n<p>\u60a8\u53ef\u80fd\u9700\u8981\u521b\u5efa\u5355\u72ec\u7684\u4efb\u52a1\u6765\u89e3\u51b3\u9488\u5bf9\u5931\u8d25\u7684\u6d4b\u8bd5\u53d1\u73b0\u7684\u9519\u8bef\uff0c\u5e76\u5c06\u5176\u94fe\u63a5\u5230\u6807\u8bc6\u76f8\u5173\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u7684\u95ee\u9898\u3002 \u60a8\u53ef\u4ee5\u4f7f\u7528\u81ea\u5b9a\u4e49\u95ee\u9898\u94fe\u63a5\u7c7b\u578b\u5c06\u5931\u8d25\u7684\u6d4b\u8bd5\u94fe\u63a5\u81f3\u5176\u76f8\u5173\u7684\u9519\u8bef\uff0c\u6216\u8005\u4f7f\u7528\u5305\u542b\u5bf9\u8be5\u9519\u8bef\u7684\u5f15\u7528\u7684\u81ea\u5b9a\u4e49\u6587\u672c\u5b57\u6bb5\u3002 \u7ec4\u7ec7\u4e00\u4e2a\u5355\u72ec\u7684 YouTrack \u9879\u76ee\u6765\u8ddf\u8e2a\u9519\u8bef\u53ef\u80fd\u66f4\u4e3a\u65b9\u4fbf\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS7.png\" alt=\"\" \/><\/p>\n<h2>\u53ef\u9884\u89c1\u7684\u529f\u80fd\u589e\u5f3a<\/h2>\n<p>\u60a8\u53ef\u4ee5\u6269\u5c55 YouTrack \u7684\u529f\u80fd\u4ee5\u6ee1\u8db3\u5176\u4ed6\u4e1a\u52a1\u9700\u6c42\u3002 \u4ee5\u4e0b\u63d0\u4f9b\u4e86\u591a\u79cd\u53ef\u4f9b\u8003\u8651\u7684\u9009\u9879\u3002<\/p>\n<h3>\u514b\u9686\u73b0\u6709\u7684\u6d4b\u8bd5\u8fd0\u884c<\/h3>\n<p>\u6709\u65f6\uff0c\u8fdb\u884c\u4e0e\u73b0\u6709\u6d4b\u8bd5\u76f8\u4f3c\u4f46\u5305\u542b\u4e00\u4e9b\u7ec6\u5fae\u53d8\u66f4\uff08\u4f8b\u5982\uff0c\u53d8\u66f4\u5206\u914d\u7684\u7248\u672c\uff09\u7684\u65b0\u6d4b\u8bd5\u8fd0\u884c\u53ef\u80fd\u4f1a\u975e\u5e38\u6709\u5e2e\u52a9\u3002 \u201cCreate Test Run copy\u201d\u83dc\u5355\u9879\u4e13\u4e3a\u6b64\u7c7b\u60c5\u51b5\u800c\u521b\u5efa\u3002 \u6b64\u83dc\u5355\u9879\u4ec5\u9002\u7528\u4e8e\u95ee\u9898\u7c7b\u578b\u4e3a\u201cTest Run\u201d\u7684\u95ee\u9898\uff0c\u53ef\u89e6\u53d1\u4ee5\u4e0b\u64cd\u4f5c\uff1a<\/p>\n<ul>\n<li>\u521b\u5efa\u6d4b\u8bd5\u8fd0\u884c\u7684\u526f\u672c\u4ee5\u53ca\u4e3a\u5176\u5206\u914d\u7684\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\u7684\u526f\u672c\u3002<\/li>\n<li>\u6240\u6709\u65b0\u521b\u5efa\u7684\u6d4b\u8bd5\u7528\u4f8b\u7684\u6267\u884c\u526f\u672c\u90fd\u4f7f\u7528\u201c\u7236-\u5b50\u201d\u95ee\u9898\u94fe\u63a5\u7c7b\u578b\u5206\u914d\u7ed9\u65b0\u6d4b\u8bd5\u8fd0\u884c\u3002 \u6b64\u5916\uff0c\u8fd9\u4e9b\u526f\u672c\u8fd8\u5c06\u4f7f\u7528\u201cExecution\u201d\u95ee\u9898\u94fe\u63a5\u7c7b\u578b\u5206\u914d\u7ed9\u539f\u59cb\u6d4b\u8bd5\u7528\u4f8b\uff08\u4ee5\u4fdd\u6301\u53ef\u8ffd\u6eaf\u6027\uff09\u3002<\/li>\n<li>\u5bf9\u4e8e\u65b0\u521b\u5efa\u7684\u95ee\u9898\uff0c\u5c06\u4e22\u5f03\u6240\u6709\u72b6\u6001\u548c\u7edf\u8ba1\u4fe1\u606f\u3002 <\/li>\n<\/ul>\n<p><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done8s720.gif\" alt=\"\" \/><\/p>\n<p>\u4ee5\u4e0a\u903b\u8f91\u4f7f\u7528\u201cCreate Test Run copy\u201d\u64cd\u4f5c\u89c4\u5219\u5b9e\u73b0\u3002<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\nvar workflow = require(&#039;@jetbrains\/youtrack-scripting-api\/workflow&#039;);\nvar entities = require(&#039;@jetbrains\/youtrack-scripting-api\/entities&#039;);\nvar utils = require(&#039;..\/calculate-tms-stats\/utils&#039;);\n\nexports.rule = entities.Issue.action({\n  title: &#039;Create Test Run copy&#039;,\n  command: &#039;Test Run Creation&#039;,\n  guard: function(ctx) {\n    return ctx.issue.isReported &amp;&amp; (ctx.issue.Type.name == ctx.Type.TestRun.name);\n  },\n  action: function(ctx) {\n    var issue = ctx.issue;\n    var TestRunCopy = issue.copy(issue.project);\n    TestRunCopy.Status = ctx.Status.InProgress; \n    var oldTestList = issue.links[ctx.Subtask.outward];\n    oldTestList.forEach(function(v) {\n      var newTest = v.copy(v.project);\n      newTest.Status = ctx.Status.InProgress;\n      newTest.links[ctx.Subtask.inward].delete(issue);\n      newTest.links[ctx.Subtask.inward].add(TestRunCopy);\n    });\n    utils.resetStatuses(issue, TestRunCopy); \n    var newTestRunLink = &#039;&lt;a href=&quot;&#039; + TestRunCopy.url + &#039;&quot;&gt; &#039; + TestRunCopy.id + &#039;&lt;\/a&gt;&#039;;\n    var message = &#039;New Test Run has been created &#039; + newTestRunLink + &#039;.&#039;;\n    workflow.message(message);\n  },\n  requirements: {\n    Execution: {\n      type: entities.IssueLinkPrototype,\n      name: &#039;Execution&#039;,\n      inward: &#039;Execution&#039;,\n      outward: &#039;Assigned Test case or test suite&#039;\n    },\n    Subtask: {\n      type: entities.IssueLinkPrototype,\n      name: &#039;Subtask&#039;,\n      inward: &#039;subtask of&#039;,\n      outward: &#039;parent for&#039;\n    },\n    Type: {\n      type: entities.EnumField.fieldType,\n      TestRun: {\n        name: &quot;Test Run&quot;\n      },\n    },\n    Status: {\n      type: entities.EnumField.fieldType,\n      InProgress: {\n        name: &#039;No Run&#039;\n      },\n    }\n  }\n});\n <\/pre>\n<\/details>\n<h2>\u9650\u5236\u7528\u6237\u7684\u64cd\u4f5c<\/h2>\n<p>\u4e3a\u4e86\u907f\u514d\u5728\u4f7f\u7528\u6d4b\u8bd5\u65b9\u6848\u65f6\u53d1\u751f\u6f5c\u5728\u9519\u8bef\uff08\u4f8b\u5982\u4f7f\u7528\u9519\u8bef\u7684\u95ee\u9898\u7c7b\u578b\uff09\uff0c\u60a8\u53ef\u4ee5\u9650\u5236\u7528\u6237\u53ef\u80fd\u91c7\u53d6\u7684\u64cd\u4f5c\u3002 \u4f8b\u5982\uff0c\u60a8\u53ef\u4ee5\u5b9e\u73b0\u64cd\u4f5c\u4ee5\u786e\u4fdd\u7528\u6237\u5c06\u6d4b\u8bd5\u7528\u4f8b\u94fe\u63a5\u5230\u6d4b\u8bd5\u8fd0\u884c\u3002\u5f53\u7528\u6237\u5c06\u6d4b\u8bd5\u8fd0\u884c\u94fe\u63a5\u5230\u4e00\u4e2a\u95ee\u9898\u65f6\uff0c\u8be5\u64cd\u4f5c\u5c06\u68c0\u67e5\u94fe\u63a5\u7684\u95ee\u9898\u7c7b\u578b\u662f \u201cTest Case\u201d \u8fd8\u662f \u201cTest Suite\u201d\u3002\u5982\u679c\u90fd\u4e0d\u662f\uff0c\u8be5\u64cd\u4f5c\u5c06\u53d1\u9001\u8b66\u544a\uff0c\u963b\u6b62\u7528\u6237\u7ee7\u7eed\u5904\u7406\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/resources.jetbrains.com\/storage\/products\/blog\/wp-content\/uploads\/done9.720.gif\" alt=\"\" \/><\/p>\n<p>\u53ef\u4ee5\u5c06\u4ee5\u4e0b\u4ee3\u7801\u5757\u6dfb\u52a0\u5230\u201cpopulate test run\u201d\u5de5\u4f5c\u6d41\u3002<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\n var message = &#039;&lt;a href=&quot;&#039; + TestCase.url + &#039;&quot;&gt; &#039; + TestCase.id + &#039;&lt;\/a&gt;&#039;;\n      workflow.check((TestCase.Type.name == ctx.Type.TestCase.name) || (TestCase.Type.name == ctx.Type.TestSuite.name), workflow.i18n(&#039;\\&#039;Test Run\\&#039; can be linked to \\&#039;Test Case\\&#039; and \\&#039;Test Suite\\&#039; only, but {0} has \\&#039;{1}\\&#039; type!&#039;, message, TestCase.Type.name));\n    <\/pre>\n<\/details>\n<p>\u5982 <a href=\"#ref2\">\u4e0a\u6587<\/a>\u6240\u8ff0\uff0c\u6d4b\u8bd5\u8fd0\u884c\u72b6\u6001\u5e94\u4e3a\u53ea\u8bfb\uff0c\u56e0\u4e3a\u5b83\u53d6\u51b3\u4e8e\u6d4b\u8bd5\u6267\u884c\u7684\u8fdb\u5ea6\u3002 \u53ea\u9700\u5305\u542b\u4e00\u4e2a\u4ee3\u7801\u5757\u5373\u53ef\u76f4\u63a5\u5b9e\u73b0\u8fd9\u4e00\u8bbe\u7f6e\uff0c\u8be5\u4ee3\u7801\u5757\u53ef\u9650\u5236\u624b\u52a8\u53d8\u66f4\u95ee\u9898\u7c7b\u578b\u4e3a\u201cTest Run\u201d\u7684\u95ee\u9898\u7684\u72b6\u6001\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/done10s720.gif\" alt=\"\" \/><br \/>\n\u4f8b\u5982\uff0c\u5728\u201cstatus-management\u201d\u5de5\u4f5c\u6d41\u4e2d\uff0c\u201cTest Run\u201d\u95ee\u9898\u7c7b\u578b\u7684\u201ctransitions\u201d\u90e8\u5206\u53ef\u4ee5\u8fdb\u884c\u4ee5\u4e0b\u8c03\u6574\uff1a<\/p>\n<details>\n<summary>\n        \u5c55\u5f00\u4ee3\u7801\u5757<br \/>\n    <\/summary>\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"js\" data-enlighter-title=\"\">\n   &#039;Failing&#039;: {\n            targetState: &#039;Failing&#039;,\n            action: function(ctx) {\n              workflow.check(false, workflow.i18n(&#039;Test run has read-only status which is defined based on assigned tests statuses&#039;));\n            }\n          },\n          ...\n        }\n<\/pre>\n<\/details>\n<h2>\u62a5\u544a\u548c\u53ef\u8ffd\u6eaf\u6027<\/h2>\n<h3>\u62a5\u544a<\/h3>\n<p>\u6700\u540e\uff0c\u5f53\u6d4b\u8bd5\u5468\u671f\u7ed3\u675f\u65f6\uff0c\u60a8\u53ef\u80fd\u9700\u8981\u5bf9\u7ed3\u679c\u8fdb\u884c\u5206\u6790\u3002 YouTrack \u62a5\u544a\u529f\u80fd\u53ef\u7528\u4e8e\u6b64\u76ee\u7684\u3002\u60a8\u53ef\u4ee5\u9009\u62e9\u62a5\u544a\u7c7b\u578b\uff0c\u5b83\u5c06\u663e\u793a\u4e0e\u6d4b\u8bd5\u7ba1\u7406\u8fc7\u7a0b\u76f8\u5173\u7684\u5173\u952e\u4fe1\u606f\u3002 \u51fa\u4e8e\u6f14\u793a\u76ee\u7684\uff0c\u6211\u4eec\u5c06\u5411\u4eea\u8868\u677f\u6dfb\u52a0 2 \u4e2a\u62a5\u544a\uff1a<\/p>\n<ul>\n<li>\n<em><b>\u6309\u201cversion\u201d\u5b57\u6bb5\u7b5b\u9009\u7684\u7d2f\u79ef\u6d41\u62a5\u544a\u3002\u6b64\u7c7b\u72b6\u6001\u62a5\u544a\u6d89\u53ca\u5230\u7279\u5b9a\u7684\u4ea7\u54c1\u7248\u672c\u3002<\/b><\/em> \u62a5\u544a\u4e2d\u663e\u793a\u4e86\u5173\u952e\u6307\u6807\uff08\u4f8b\u5982\u72b6\u6001\u4e3a\u201cPassed\u201d\u3001\u201cFailed\u201d\u548c\u201cNo Run\u201d\u7684\u6d4b\u8bd5\u7684\u6570\u91cf\uff09\uff0c\u5e76\u5728\u65f6\u95f4\u7ebf\u4e0a\u5448\u73b0\u3002 \u8981\u663e\u793a\u8de8\u7248\u672c\u6570\u636e\uff0c\u60a8\u5e94\u8be5\u4e3a\u6bcf\u4e2a\u7248\u672c\u521b\u5efa\u4e00\u4e2a\u62a5\u544a\u3002 \u5efa\u8bae\u5c06\u6240\u6709\u62a5\u544a\u6dfb\u52a0\u5230\u4eea\u8868\u677f\u4e2d\uff0c\u8fd9\u79cd\u65b9\u5f0f\u53ef\u4ee5\u5c06\u6240\u6709\u6570\u636e\u96c6\u4e2d\u5230\u4e00\u5904\uff0c\u4fbf\u4e8e\u4f7f\u7528\u3002<\/p>\n<h3>\u62a5\u544a\u8bbe\u7f6e\uff1a<\/h3>\n<p><a href=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS11.png\" rel=\"attachment wp-att-6366\"><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS11.png\" alt=\"ApplyingCommandForSubsystem\" \/><\/a><\/p>\n<h3>\u62a5\u544a\u7ed3\u679c\uff1a<\/h3>\n<p><a href=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS12.png\" rel=\"attachment wp-att-6366\"><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS12.png\" alt=\"ApplyingCommandForSubsystem\" \/><\/a><\/p>\n<\/li>\n<li><em> <b>\u95ee\u9898\u5206\u5e03\u62a5\u544a\u3002<\/b><\/em> \u8be5\u62a5\u544a\u662f\u4e00\u5f20\u5feb\u7167\uff0c\u5176\u4e2d\u5305\u542b\u591a\u4e2a\u4ea7\u54c1\u7248\u672c\u7684\u5173\u952e\u6307\u6807\uff08\u4f8b\u5982\uff0c\u6bcf\u79cd\u72b6\u6001\u7684\u6d4b\u8bd5\u8fd0\u884c\u7684\u6570\u91cf\uff09\u3002 \u6211\u4eec\u4e13\u95e8\u9488\u5bf9\u6b64\u6f14\u793a\u63d0\u4f9b\u4e86\u62a5\u544a\u7ed3\u679c\uff0c\u53ef\u5e2e\u52a9\u60a8\u5bf9\u6bd4\u5404\u4e2a\u7248\u672c\u5e76\u5206\u6790\u7248\u672c\u7a33\u5b9a\u6027\u3002<br \/>\n<h3>\u62a5\u544a\u8bbe\u7f6e\uff1a<\/h3>\n<p><a href=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS13.png\" rel=\"attachment wp-att-6366\"><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS13.png\" alt=\"ApplyingCommandForSubsystem\" \/><\/a><\/p>\n<h3>\u62a5\u544a\u7ed3\u679c\uff1a<\/h3>\n<p><a href=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/re_res_cut.png\" rel=\"attachment wp-att-6366\"><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/re_res_cut.png\" alt=\"ApplyingCommandForSubsystem\" \/><\/a>\n <\/li>\n<\/ul>\n<h2>\u53ef\u8ffd\u6eaf\u6027<\/h2>\n<p>YouTrack \u4f1a\u9488\u5bf9\u6bcf\u4e2a\u201cTest Execution\u201d\u7c7b\u578b\u7684\u95ee\u9898\u5b58\u50a8\u4e0e\u6d4b\u8bd5\u5468\u671f\u76f8\u5173\u7684\u5173\u952e\u4fe1\u606f\uff0c\u56e0\u6b64\u60a8\u59cb\u7ec8\u53ef\u4ee5\u8bbf\u95ee\u6d4b\u8bd5\u5386\u53f2\u8bb0\u5f55\u5e76\u8bc6\u522b\u4efb\u4f55\u76f8\u5173\u9519\u8bef\u3002<br \/>\n\u5982\u679c\u60a8\u6709\u4e00\u4e2a\u6d4b\u8bd5\u7528\u4f8b\uff0c\u5219\u53ef\u4ee5\u5f15\u7528\u4efb\u4f55\u6d89\u53ca\u8be5\u6d4b\u8bd5\u7528\u4f8b\u7684\u6d4b\u8bd5\u6267\u884c\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS15.png\" alt=\"\" \/><\/p>\n<p>\u5bf9\u4e8e\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\uff0c\u60a8\u53ef\u4ee5\u5f15\u7528\u6d4b\u8bd5\u8fd0\u884c\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS16.png\" alt=\"\" \/><\/p>\n<p>\u9488\u5bf9\u5931\u8d25\u7684\u6d4b\u8bd5\u7528\u4f8b\u6267\u884c\uff0c\u60a8\u59cb\u7ec8\u53ef\u4ee5\u5f15\u7528\u76f8\u5173 Bug \u5217\u8868\u3002<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/blog.jetbrains.com\/wp-content\/uploads\/2020\/09\/TMS17.png\" alt=\"\" \/><\/p>\n<p>\u8981\u8be6\u7ec6\u4e86\u89e3\u672c\u6587\u6240\u8ff0\u5de5\u4f5c\u6d41\u6216\u8ba8\u8bba\u4e0e\u5de5\u4f5c\u6d41\u76f8\u5173\u7684\u5176\u4ed6\u4e3b\u9898\uff0c\u8bf7\u968f\u65f6\u52a0\u5165\u6211\u4eec\u5728 Slack \u4e2d\u7684 <a href=\"http:\/\/youtrack-community.herokuapp.com\/\" target=\"_blank\" rel=\"noopener\">YouTrack \u5de5\u4f5c\u6d41\u793e\u533a<\/a>\u3002<\/p>\n","protected":false},"author":814,"featured_media":0,"comment_status":"closed","ping_status":"closed","template":"","categories":[],"tags":[],"cross-post-tag":[],"acf":[],"_links":{"self":[{"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/youtrack\/95720"}],"collection":[{"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/youtrack"}],"about":[{"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/types\/youtrack"}],"author":[{"embeddable":true,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/users\/814"}],"replies":[{"embeddable":true,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/comments?post=95720"}],"version-history":[{"count":1,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/youtrack\/95720\/revisions"}],"predecessor-version":[{"id":95994,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/youtrack\/95720\/revisions\/95994"}],"wp:attachment":[{"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/media?parent=95720"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/categories?post=95720"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/tags?post=95720"},{"taxonomy":"cross-post-tag","embeddable":true,"href":"https:\/\/blog.jetbrains.com\/zh-hans\/wp-json\/wp\/v2\/cross-post-tag?post=95720"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}