diff --git a/docs/pipelines.md b/docs/pipelines.md index b6ae4552844..afeb65a92fc 100644 --- a/docs/pipelines.md +++ b/docs/pipelines.md @@ -25,6 +25,7 @@ weight: 3 - [Configuring the `Task` execution order](#configuring-the-task-execution-order) - [Adding a description](#adding-a-description) - [Adding `Finally` to the `Pipeline`](#adding-finally-to-the-pipeline) + - [Using Custom Tasks](#using-custom-tasks) - [Code examples](#code-examples) ## Overview @@ -885,6 +886,87 @@ In this example, `PipelineResults` is set to: ], ``` +## Using Custom Tasks + +[Custom Tasks](https://github.com/tektoncd/community/blob/master/teps/0002-custom-tasks.md) +can implement behavior that doesn't correspond directly to running a workload in a `Pod` on the cluster. +For example, a custom task might execute some operation outside of the cluster and wait for its execution to complete. + +A PipelineRun starts a custom task by creating a [`Run`](https://github.com/tektoncd/pipeline/blob/master/docs/runs.md) instead of a `TaskRun`. +In order for a custom task to execute, there must be a custom task controller running on the cluster +that is responsible for watching and updating `Run`s which reference their type. +If no such controller is running, those `Run`s will never complete and Pipelines using them will time out. + +Custom tasks are an **_experimental alpha feature_** and should be expected to change +in breaking ways or even be removed. + +### Specifying the target Custom Task + +To specify the custom task type you want to execute, the `taskRef` field +must include the custom task's `apiVersion` and `kind` as shown below: + +```yaml +spec: + tasks: + - name: run-custom-task + taskRef: + apiVersion: example.dev/v1alpha1 + kind: Example +``` + +This creates a `Run` of a custom task of type `Example` in the `example.dev` API group with the version `v1alpha1`. + +You can also specify the `name` of a custom task resource object previously defined in the cluster. + +```yaml +spec: + tasks: + - name: run-custom-task + taskRef: + apiVersion: example.dev/v1alpha1 + kind: Example + name: myexample +``` + +If the `taskRef` specifies a name, the custom task controller should look up the +`Example` resource with that name and use that object to configure the execution. + +If the `taskRef` does not specify a name, the custom task controller might support +some default behavior for executing unnamed tasks. + +### Specifying `Parameters` + +If a custom task supports [`parameters`](tasks.md#parameters), you can use the +`params` field to specify their values: + +```yaml +spec: +spec: + tasks: + - name: run-custom-task + taskRef: + apiVersion: example.dev/v1alpha1 + kind: Example + name: myexample + params: + - name: foo + value: bah +``` + +### Using `Results` + +If the custom task produces results, you can reference them in a Pipeline using the normal syntax, +`$(tasks..results.)`. + +### Limitations + +Pipelines do not directly support passing the following items to custom tasks: +* Pipeline Resources +* Workspaces +* Service account name +* Pod templates + + ## Code examples For a better understanding of `Pipelines`, study [our code examples](https://github.com/tektoncd/pipeline/tree/master/examples). diff --git a/docs/runs.md b/docs/runs.md index 6800cb65440..64868050a17 100644 --- a/docs/runs.md +++ b/docs/runs.md @@ -35,10 +35,6 @@ will have no `.status` value and no further action will be taken. `Run`s are an **_experimental alpha feature_** and should be expected to change in breaking ways or even be removed. -`Run`s are not currently integrated with Pipelines, and require a running -third-party controller to actually perform any work. Without a third-party -controller, `Run`s will just exist without a status indefinitely. - ## Configuring a `Run` A `Run` definition supports the following fields: diff --git a/go.sum b/go.sum index ee1e4de4323..1e18d3d927c 100644 --- a/go.sum +++ b/go.sum @@ -343,8 +343,8 @@ github.com/docker/cli v0.0.0-20190925022749-754388324470/go.mod h1:JLrzqnKDaYBop github.com/docker/cli v0.0.0-20191017083524-a8ff7f821017 h1:2HQmlpI3yI9deH18Q6xiSOIjXD4sLI55Y/gfpa8/558= github.com/docker/cli v0.0.0-20191017083524-a8ff7f821017/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= github.com/docker/cli v0.0.0-20200130152716-5d0cf8839492/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= -github.com/docker/cli v0.0.0-20200303162255-7d407207c304 h1:A7SYzidcyuQ/yS4wezWGYeUioUFJQk8HYWY9aMYTF4I= -github.com/docker/cli v0.0.0-20200303162255-7d407207c304/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= +github.com/docker/cli v0.0.0-20200210162036-a4bedce16568 h1:AbI1uj9w4yt6TvfKHfRu7G55KuQe7NCvWPQRKDoXggE= +github.com/docker/cli v0.0.0-20200210162036-a4bedce16568/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8= github.com/docker/distribution v0.0.0-20191216044856-a8371794149d/go.mod h1:0+TTO4EOBfRPhZXAeF1Vu+W3hHZ8eLp8PgKVZlcvtFY= github.com/docker/distribution v2.6.0-rc.1.0.20180327202408-83389a148052+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= github.com/docker/distribution v2.7.1+incompatible h1:a5mlkVzth6W5A4fOsS3D2EO5BUmsJpcB+cRlLU7cSug= @@ -636,7 +636,9 @@ github.com/google/licenseclassifier v0.0.0-20200402202327-879cb1424de0/go.mod h1 github.com/google/licenseclassifier v0.0.0-20200708223521-3d09a0ea2f39/go.mod h1:qsqn2hxC+vURpyBRygGUuinTO42MFRLcsmQ/P8v94+M= github.com/google/mako v0.0.0-20190821191249-122f8dcef9e3/go.mod h1:YzLcVlL+NqWnmUEPuhS1LxDDwGO9WNbVlEXaF4IH35g= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/martian v2.1.1-0.20190517191504-25dcb96d9e51+incompatible h1:xmapqc1AyLoB+ddYT6r04bD9lIjlOqGaREovi0SzFaE= github.com/google/martian v2.1.1-0.20190517191504-25dcb96d9e51+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/martian/v3 v3.0.0 h1:pMen7vLs8nvgEYhywH3KDWJIJTeEr2ULsVWHWYHQyBs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= @@ -830,6 +832,7 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/pty v1.1.5/go.mod h1:9r2w37qlBe7rQ6e1fg1S/9xpWHSnaqNdHD3WcMdbPDA= github.com/kr/pty v1.1.8/go.mod h1:O1sed60cT9XZ5uDucP5qwvh+TE3NnUj51EiZO/lmSfw= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/lib/pq v1.0.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= github.com/lib/pq v1.1.1/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= @@ -934,11 +937,14 @@ github.com/nats-io/nkeys v0.1.0/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxzi github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= github.com/nats-io/nuid v1.0.0/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= +github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32 h1:W6apQkHrMkS0Muv8G/TipAy/FJl/rCYT0+EuS8+Z0z4= github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32/go.mod h1:9wM+0iRr9ahx58uYLpLIr5fm8diHn0JbqRycJi6w0Ms= github.com/nbutton23/zxcvbn-go v0.0.0-20180912185939-ae427f1e4c1d/go.mod h1:o96djdrsSGy3AWPyBgZMAGfxZNfgntdJG+11KU4QvbU= github.com/ncw/swift v1.0.47/go.mod h1:23YIA4yWVnGwv2dQlN4bB7egfYX6YLn0Yo/S6zZO/ZM= +github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/nwaples/rardecode v1.0.0/go.mod h1:5DzqNKiOdpKKBH87u8VlvAnPZMXcGRhxWkRpHbbfGS0= +github.com/nxadm/tail v1.4.4 h1:DQuhQpB1tVlglWS2hLQ5OV6B5r8aGxSrPc5Qo6uTN78= github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= github.com/octago/sflags v0.2.0/go.mod h1:G0bjdxh4qPRycF74a2B8pU36iTp9QHGx0w0dFZXPt80= github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U= @@ -951,6 +957,7 @@ github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+ github.com/onsi/ginkgo v1.10.2/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.12.0/go.mod h1:oUhWkIvk5aDxtKvDDuw8gItl8pKl42LzjC9KZE0HfGg= +github.com/onsi/ginkgo v1.12.1 h1:mFwc4LvZ0xpSvDZ3E+k8Yte0hLOMxXUlP+yXtJqkYfQ= github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA= github.com/onsi/gomega v1.4.2/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= @@ -960,6 +967,7 @@ github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1Cpa github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= github.com/onsi/gomega v1.8.1/go.mod h1:Ho0h+IUsWyvy1OpqCwxlQ/21gkhVunqlU8fDGcoTdcA= github.com/onsi/gomega v1.9.0/go.mod h1:Ho0h+IUsWyvy1OpqCwxlQ/21gkhVunqlU8fDGcoTdcA= +github.com/onsi/gomega v1.10.1 h1:o0+MgICZLuZ7xjH7Vx6zS/zcu93/BEp1VwkIW1mEXCE= github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= github.com/opencontainers/go-digest v0.0.0-20170106003457-a6d0ee40d420/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s= @@ -1001,6 +1009,7 @@ github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINE github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/profile v1.2.1/go.mod h1:hJw3o1OdXxsrSjjVksARp5W95eeEaEfptyVZyv6JUPA= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= github.com/pquerna/cachecontrol v0.0.0-20171018203845-0dec1b30a021/go.mod h1:prYjPmNq4d1NPVmpShWobRqXY3q7Vp+80DqgxxUrUIA= @@ -1152,6 +1161,7 @@ github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoH github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= +github.com/stretchr/testify v1.5.1 h1:nOGnQDM7FYENwehXlg/kFVnos3rEvtKTjRvOWSzb6H4= github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/subosito/gotenv v1.2.0 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s= github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= @@ -1192,6 +1202,7 @@ github.com/urfave/cli v0.0.0-20171014202726-7bc6a0acffa5/go.mod h1:70zkFmudgCuE/ github.com/urfave/cli v1.18.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= github.com/uudashr/gocognit v1.0.1/go.mod h1:j44Ayx2KW4+oB6SWMv8KsmHzZrOInQav7D3cQMJ5JUM= +github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= github.com/valyala/fasthttp v1.2.0/go.mod h1:4vX61m6KN+xDduDNwXrhIAVZaZaZiQ1luJk8LWSxF3s= github.com/valyala/quicktemplate v1.2.0/go.mod h1:EH+4AkTd43SvgIbQHYu59/cJyxDoOVRUAfrukLPuGJ4= @@ -1256,6 +1267,7 @@ go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+ go.uber.org/multierr v1.4.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= go.uber.org/multierr v1.5.0 h1:KCa4XfM8CWFCpxXRGok+Q0SS/0XBhMDbHHGABQLvD2A= go.uber.org/multierr v1.5.0/go.mod h1:FeouvMocqHpRaaGuG9EjoKcStLC43Zu/fmqdUMPcKYU= +go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee h1:0mgffUl7nfd+FpvXMVz4IDEaUSmT1ysygQC7qYo7sG4= go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= go.uber.org/zap v1.9.1/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= go.uber.org/zap v1.9.2-0.20180814183419-67bc79d13d15/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= @@ -1523,7 +1535,6 @@ golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgw golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190617190820-da514acc4774/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190624222133-a101b041ded4/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190706070813-72ffa07ba3db/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI= golang.org/x/tools v0.0.0-20190719005602-e377ae9d6386/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI= @@ -1746,6 +1757,7 @@ gopkg.in/check.v1 v1.0.0-20141024133853-64131543e789/go.mod h1:Co6ibVJAznAaIkqp8 gopkg.in/check.v1 v1.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= @@ -1771,6 +1783,7 @@ gopkg.in/square/go-jose.v2 v2.2.2/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76 gopkg.in/src-d/go-billy.v4 v4.3.2/go.mod h1:nDjArDMp+XMs1aFAESLRjfGSgfvoYN0hDfzEk0GjC98= gopkg.in/src-d/go-git-fixtures.v3 v3.5.0/go.mod h1:dLBcvytrw/TYZsNTWCnkNF2DSIlzWYqTe3rJR56Ac7g= gopkg.in/src-d/go-git.v4 v4.13.1/go.mod h1:nx5NYcxdKxq5fpltdHnPa2Exj4Sx0EclMWZQbYDu2z8= +gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/warnings.v0 v0.1.1/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= @@ -1785,9 +1798,8 @@ gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20190709130402-674ba3eaed22/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw= -gotest.tools/v3 v3.0.2 h1:kG1BFyqVHuQoVQiR1bWGnfz/fmHvvuiSPIV7rvl360E= -gotest.tools/v3 v3.0.2/go.mod h1:3SzNCllyD9/Y+b5r9JIKQ474KzkZyqLqEfYqMsX94Bk= helm.sh/helm/v3 v3.1.1/go.mod h1:WYsFJuMASa/4XUqLyv54s0U/f3mlAaRErGmyy4z921g= honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= diff --git a/hack/update-codegen.sh b/hack/update-codegen.sh index 022a7111fe3..7ccfaa0279f 100755 --- a/hack/update-codegen.sh +++ b/hack/update-codegen.sh @@ -58,6 +58,11 @@ ${PREFIX}/deepcopy-gen \ --go-header-file ${REPO_ROOT_DIR}/hack/boilerplate/boilerplate.go.txt \ -i github.com/tektoncd/pipeline/pkg/apis/resource/v1alpha1/storage +${PREFIX}/deepcopy-gen \ + -O zz_generated.deepcopy \ + --go-header-file ${REPO_ROOT_DIR}/hack/boilerplate/boilerplate.go.txt \ +-i github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1 + # Knative Injection # This generates the knative injection packages for the resource package (v1alpha1). # This is separate from the pipeline package for the same reason as client and all (see above). diff --git a/pkg/apis/pipeline/v1alpha1/run_types.go b/pkg/apis/pipeline/v1alpha1/run_types.go index 9f07eb4ba72..6cb820ef5b6 100644 --- a/pkg/apis/pipeline/v1alpha1/run_types.go +++ b/pkg/apis/pipeline/v1alpha1/run_types.go @@ -18,14 +18,12 @@ package v1alpha1 import ( "fmt" - "time" "github.com/tektoncd/pipeline/pkg/apis/pipeline" v1beta1 "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" + runv1alpha1 "github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" - "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime/schema" - "k8s.io/apimachinery/pkg/util/json" "knative.dev/pkg/apis" duckv1 "knative.dev/pkg/apis/duck/v1" ) @@ -75,85 +73,16 @@ func (rs RunSpec) GetParam(name string) *v1beta1.Param { return nil } -type RunStatus struct { - duckv1.Status `json:",inline"` +const ( + // RunReasonCancelled must be used in the Condition Reason to indicate that a Run was cancelled. + RunReasonCancelled = "RunCancelled" +) - // RunStatusFields inlines the status fields. - RunStatusFields `json:",inline"` -} +// RunStatus defines the observed state of Run. +type RunStatus = runv1alpha1.RunStatus var runCondSet = apis.NewBatchConditionSet() -// GetCondition returns the Condition matching the given type. -func (r *RunStatus) GetCondition(t apis.ConditionType) *apis.Condition { - return runCondSet.Manage(r).GetCondition(t) -} - -// InitializeConditions will set all conditions in runCondSet to unknown for the PipelineRun -// and set the started time to the current time -func (r *RunStatus) InitializeConditions() { - started := false - if r.StartTime.IsZero() { - r.StartTime = &metav1.Time{Time: time.Now()} - started = true - } - conditionManager := runCondSet.Manage(r) - conditionManager.InitializeConditions() - // Ensure the started reason is set for the "Succeeded" condition - if started { - initialCondition := conditionManager.GetCondition(apis.ConditionSucceeded) - initialCondition.Reason = "Started" - conditionManager.SetCondition(*initialCondition) - } -} - -// SetCondition sets the condition, unsetting previous conditions with the same -// type as necessary. -func (r *RunStatus) SetCondition(newCond *apis.Condition) { - if newCond != nil { - runCondSet.Manage(r).SetCondition(*newCond) - } -} - -// MarkRunSucceeded changes the Succeeded condition to True with the provided reason and message. -func (r *RunStatus) MarkRunSucceeded(reason, messageFormat string, messageA ...interface{}) { - runCondSet.Manage(r).MarkTrueWithReason(apis.ConditionSucceeded, reason, messageFormat, messageA...) - succeeded := r.GetCondition(apis.ConditionSucceeded) - r.CompletionTime = &succeeded.LastTransitionTime.Inner -} - -// MarkRunFailed changes the Succeeded condition to False with the provided reason and message. -func (r *RunStatus) MarkRunFailed(reason, messageFormat string, messageA ...interface{}) { - runCondSet.Manage(r).MarkFalse(apis.ConditionSucceeded, reason, messageFormat, messageA...) - succeeded := r.GetCondition(apis.ConditionSucceeded) - r.CompletionTime = &succeeded.LastTransitionTime.Inner -} - -// MarkRunRunning changes the Succeeded condition to Unknown with the provided reason and message. -func (r *RunStatus) MarkRunRunning(reason, messageFormat string, messageA ...interface{}) { - runCondSet.Manage(r).MarkUnknown(apis.ConditionSucceeded, reason, messageFormat, messageA...) -} - -// DecodeExtraFields deserializes the extra fields in the Run status. -func (r *RunStatus) DecodeExtraFields(into interface{}) error { - if len(r.ExtraFields.Raw) == 0 { - return nil - } - return json.Unmarshal(r.ExtraFields.Raw, into) -} - -// EncodeExtraFields serializes the extra fields in the Run status. -func (r *RunStatus) EncodeExtraFields(from interface{}) error { - data, err := json.Marshal(from) - if err != nil { - return err - } - r.ExtraFields = runtime.RawExtension{ - Raw: data, - } - return nil -} - // GetConditionSet retrieves the condition set for this resource. Implements // the KRShaped interface. func (r *Run) GetConditionSet() apis.ConditionSet { return runCondSet } @@ -165,24 +94,10 @@ func (r *Run) GetStatus() *duckv1.Status { return &r.Status.Status } // RunStatusFields holds the fields of Run's status. This is defined // separately and inlined so that other types can readily consume these fields // via duck typing. -type RunStatusFields struct { - // StartTime is the time the build is actually started. - // +optional - StartTime *metav1.Time `json:"startTime,omitempty"` - - // CompletionTime is the time the build completed. - // +optional - CompletionTime *metav1.Time `json:"completionTime,omitempty"` - - // Results reports any output result values to be consumed by later - // tasks in a pipeline. - // +optional - Results []v1beta1.TaskRunResult `json:"results,omitempty"` +type RunStatusFields = runv1alpha1.RunStatusFields - // ExtraFields holds arbitrary fields provided by the custom task - // controller. - ExtraFields runtime.RawExtension `json:"extraFields,omitempty"` -} +// RunResult used to describe the results of a task +type RunResult = runv1alpha1.RunResult // +genclient // +genreconciler diff --git a/pkg/apis/pipeline/v1alpha1/run_types_test.go b/pkg/apis/pipeline/v1alpha1/run_types_test.go index 4e713fbe13a..fd4280a0f61 100644 --- a/pkg/apis/pipeline/v1alpha1/run_types_test.go +++ b/pkg/apis/pipeline/v1alpha1/run_types_test.go @@ -18,6 +18,7 @@ package v1alpha1_test import ( "testing" + "time" "github.com/google/go-cmp/cmp" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" @@ -102,6 +103,71 @@ func TestGetParams(t *testing.T) { } } +func TestRunHasStarted(t *testing.T) { + params := []struct { + name string + runStatus v1alpha1.RunStatus + expectedValue bool + }{{ + name: "runWithNoStartTime", + runStatus: v1alpha1.RunStatus{}, + expectedValue: false, + }, { + name: "runWithStartTime", + runStatus: v1alpha1.RunStatus{ + RunStatusFields: v1alpha1.RunStatusFields{ + StartTime: &metav1.Time{Time: time.Now()}, + }, + }, + expectedValue: true, + }, { + name: "runWithZeroStartTime", + runStatus: v1alpha1.RunStatus{ + RunStatusFields: v1alpha1.RunStatusFields{ + StartTime: &metav1.Time{}, + }, + }, + expectedValue: false, + }} + for _, tc := range params { + t.Run(tc.name, func(t *testing.T) { + run := v1alpha1.Run{} + run.Status = tc.runStatus + if run.HasStarted() != tc.expectedValue { + t.Fatalf("Expected run HasStarted() to return %t but got %t", tc.expectedValue, run.HasStarted()) + } + }) + } +} + +func TestRunIsDone(t *testing.T) { + run := v1alpha1.Run{ + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: duckv1.Conditions{{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionFalse, + }}, + }, + }, + } + + if !run.IsDone() { + t.Fatal("Expected run status to be done") + } +} + +func TestRunIsCancelled(t *testing.T) { + run := v1alpha1.Run{ + Spec: v1alpha1.RunSpec{ + Status: v1alpha1.RunSpecStatusCancelled, + }, + } + if !run.IsCancelled() { + t.Fatal("Expected run status to be cancelled") + } +} + // TestRunStatusExtraFields tests that extraFields in a RunStatus can be parsed // from YAML. func TestRunStatus(t *testing.T) { @@ -153,7 +219,7 @@ status: }, RunStatusFields: v1alpha1.RunStatusFields{ // Results are parsed correctly. - Results: []v1beta1.TaskRunResult{{ + Results: []v1alpha1.RunResult{{ Name: "foo", Value: "bar", }}, diff --git a/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go index cf2dbdfb958..b0b06c95bcf 100644 --- a/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1alpha1/zz_generated.deepcopy.go @@ -699,54 +699,6 @@ func (in *RunSpec) DeepCopy() *RunSpec { return out } -// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. -func (in *RunStatus) DeepCopyInto(out *RunStatus) { - *out = *in - in.Status.DeepCopyInto(&out.Status) - in.RunStatusFields.DeepCopyInto(&out.RunStatusFields) - return -} - -// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunStatus. -func (in *RunStatus) DeepCopy() *RunStatus { - if in == nil { - return nil - } - out := new(RunStatus) - in.DeepCopyInto(out) - return out -} - -// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. -func (in *RunStatusFields) DeepCopyInto(out *RunStatusFields) { - *out = *in - if in.StartTime != nil { - in, out := &in.StartTime, &out.StartTime - *out = (*in).DeepCopy() - } - if in.CompletionTime != nil { - in, out := &in.CompletionTime, &out.CompletionTime - *out = (*in).DeepCopy() - } - if in.Results != nil { - in, out := &in.Results, &out.Results - *out = make([]v1beta1.TaskRunResult, len(*in)) - copy(*out, *in) - } - in.ExtraFields.DeepCopyInto(&out.ExtraFields) - return -} - -// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunStatusFields. -func (in *RunStatusFields) DeepCopy() *RunStatusFields { - if in == nil { - return nil - } - out := new(RunStatusFields) - in.DeepCopyInto(out) - return out -} - // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *Task) DeepCopyInto(out *Task) { *out = *in diff --git a/pkg/apis/pipeline/v1beta1/openapi_generated.go b/pkg/apis/pipeline/v1beta1/openapi_generated.go index 7ff7525a0cb..e06d6844bd2 100644 --- a/pkg/apis/pipeline/v1beta1/openapi_generated.go +++ b/pkg/apis/pipeline/v1beta1/openapi_generated.go @@ -55,6 +55,7 @@ func GetOpenAPIDefinitions(ref common.ReferenceCallback) map[string]common.OpenA "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunConditionCheckStatus": schema_pkg_apis_pipeline_v1beta1_PipelineRunConditionCheckStatus(ref), "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunList": schema_pkg_apis_pipeline_v1beta1_PipelineRunList(ref), "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunResult": schema_pkg_apis_pipeline_v1beta1_PipelineRunResult(ref), + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunRunStatus": schema_pkg_apis_pipeline_v1beta1_PipelineRunRunStatus(ref), "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunSpec": schema_pkg_apis_pipeline_v1beta1_PipelineRunSpec(ref), "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunSpecServiceAccountName": schema_pkg_apis_pipeline_v1beta1_PipelineRunSpecServiceAccountName(ref), "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunStatus": schema_pkg_apis_pipeline_v1beta1_PipelineRunStatus(ref), @@ -1322,6 +1323,47 @@ func schema_pkg_apis_pipeline_v1beta1_PipelineRunResult(ref common.ReferenceCall } } +func schema_pkg_apis_pipeline_v1beta1_PipelineRunRunStatus(ref common.ReferenceCallback) common.OpenAPIDefinition { + return common.OpenAPIDefinition{ + Schema: spec.Schema{ + SchemaProps: spec.SchemaProps{ + Description: "PipelineRunRunStatus contains the name of the PipelineTask for this Run and the Run's Status", + Type: []string{"object"}, + Properties: map[string]spec.Schema{ + "pipelineTaskName": { + SchemaProps: spec.SchemaProps{ + Description: "PipelineTaskName is the name of the PipelineTask.", + Type: []string{"string"}, + Format: "", + }, + }, + "status": { + SchemaProps: spec.SchemaProps{ + Description: "Status is the RunStatus for the corresponding Run", + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1.RunStatus"), + }, + }, + "whenExpressions": { + SchemaProps: spec.SchemaProps{ + Description: "WhenExpressions is the list of checks guarding the execution of the PipelineTask", + Type: []string{"array"}, + Items: &spec.SchemaOrArray{ + Schema: &spec.Schema{ + SchemaProps: spec.SchemaProps{ + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.WhenExpression"), + }, + }, + }, + }, + }, + }, + }, + }, + Dependencies: []string{ + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.WhenExpression", "github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1.RunStatus"}, + } +} + func schema_pkg_apis_pipeline_v1beta1_PipelineRunSpec(ref common.ReferenceCallback) common.OpenAPIDefinition { return common.OpenAPIDefinition{ Schema: spec.Schema{ @@ -1536,6 +1578,20 @@ func schema_pkg_apis_pipeline_v1beta1_PipelineRunStatus(ref common.ReferenceCall }, }, }, + "runs": { + SchemaProps: spec.SchemaProps{ + Description: "map of PipelineRunRunStatus with the run name as the key", + Type: []string{"object"}, + AdditionalProperties: &spec.SchemaOrBool{ + Allows: true, + Schema: &spec.Schema{ + SchemaProps: spec.SchemaProps{ + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunRunStatus"), + }, + }, + }, + }, + }, "pipelineResults": { SchemaProps: spec.SchemaProps{ Description: "PipelineResults are the list of results written out by the pipeline task's containers", @@ -1572,7 +1628,7 @@ func schema_pkg_apis_pipeline_v1beta1_PipelineRunStatus(ref common.ReferenceCall }, }, Dependencies: []string{ - "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunResult", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunTaskRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineSpec", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.SkippedTask", "k8s.io/apimachinery/pkg/apis/meta/v1.Time", "knative.dev/pkg/apis.Condition"}, + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunResult", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunTaskRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineSpec", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.SkippedTask", "k8s.io/apimachinery/pkg/apis/meta/v1.Time", "knative.dev/pkg/apis.Condition"}, } } @@ -1609,6 +1665,20 @@ func schema_pkg_apis_pipeline_v1beta1_PipelineRunStatusFields(ref common.Referen }, }, }, + "runs": { + SchemaProps: spec.SchemaProps{ + Description: "map of PipelineRunRunStatus with the run name as the key", + Type: []string{"object"}, + AdditionalProperties: &spec.SchemaOrBool{ + Allows: true, + Schema: &spec.Schema{ + SchemaProps: spec.SchemaProps{ + Ref: ref("github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunRunStatus"), + }, + }, + }, + }, + }, "pipelineResults": { SchemaProps: spec.SchemaProps{ Description: "PipelineResults are the list of results written out by the pipeline task's containers", @@ -1645,7 +1715,7 @@ func schema_pkg_apis_pipeline_v1beta1_PipelineRunStatusFields(ref common.Referen }, }, Dependencies: []string{ - "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunResult", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunTaskRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineSpec", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.SkippedTask", "k8s.io/apimachinery/pkg/apis/meta/v1.Time"}, + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunResult", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineRunTaskRunStatus", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.PipelineSpec", "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1.SkippedTask", "k8s.io/apimachinery/pkg/apis/meta/v1.Time"}, } } diff --git a/pkg/apis/pipeline/v1beta1/pipeline_validation.go b/pkg/apis/pipeline/v1beta1/pipeline_validation.go index d10d75b1617..0fcc2c2c498 100644 --- a/pkg/apis/pipeline/v1beta1/pipeline_validation.go +++ b/pkg/apis/pipeline/v1beta1/pipeline_validation.go @@ -132,25 +132,25 @@ func validatePipelineTask(ctx context.Context, t PipelineTask, taskNames sets.St } if hasTaskRef && !isCustomTask && t.TaskRef.Name == "" { - return apis.ErrInvalidValue(t.TaskRef, "taskRef must specify name") + errs = errs.Also(apis.ErrInvalidValue("taskRef must specify name", "taskRef.name")) } if isCustomTask && t.TaskRef.Kind == "" { - return apis.ErrInvalidValue(t.TaskRef, "custom task ref must specify apiVersion and kind") + errs = errs.Also(apis.ErrInvalidValue("custom task ref must specify kind", "taskRef.kind")) } // TODO(#3133): Support these features if possible. if isCustomTask { if t.Retries > 0 { - return apis.ErrInvalidValue(t.Retries, "custom tasks do not support Retries") + errs = errs.Also(apis.ErrInvalidValue("custom tasks do not support retries", "retries")) } if t.Resources != nil { - return apis.ErrInvalidValue(t.Resources, "custom tasks do not support PipelineResources") + errs = errs.Also(apis.ErrInvalidValue("custom tasks do not support PipelineResources", "resources")) } if len(t.Workspaces) > 0 { - return apis.ErrInvalidValue(t.Workspaces, "custom tasks do not support Workspaces") + errs = errs.Also(apis.ErrInvalidValue("custom tasks do not support Workspaces", "workspaces")) } if t.Timeout != nil { - return apis.ErrInvalidValue(t.Timeout, "custom tasks do not support Timeout") + errs = errs.Also(apis.ErrInvalidValue("custom tasks do not support timeout", "timeout")) } } diff --git a/pkg/apis/pipeline/v1beta1/pipeline_validation_test.go b/pkg/apis/pipeline/v1beta1/pipeline_validation_test.go index cf3a2c7e3b0..7f3b5425cf1 100644 --- a/pkg/apis/pipeline/v1beta1/pipeline_validation_test.go +++ b/pkg/apis/pipeline/v1beta1/pipeline_validation_test.go @@ -526,15 +526,15 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { name: "pipelinetask taskRef without name", tasks: []PipelineTask{{Name: "foo", TaskRef: &TaskRef{Name: ""}}}, expectedError: apis.FieldError{ - Message: `GMD FIX ME`, - Paths: []string{"tasks[0].taskSpec.steps"}, + Message: `invalid value: taskRef must specify name`, + Paths: []string{"tasks[0].taskRef.name"}, }, }, { - name: "pipelinetask custom task taskRef without name", + name: "pipelinetask custom task taskRef without kind", tasks: []PipelineTask{{Name: "foo", TaskRef: &TaskRef{APIVersion: "example.dev/v0", Kind: "", Name: ""}}}, expectedError: apis.FieldError{ - Message: `GMD FIX ME -- i think this one is not an error though`, - Paths: []string{"tasks[0].taskSpec.steps"}, + Message: `invalid value: custom task ref must specify kind`, + Paths: []string{"tasks[0].taskRef.kind"}, }, }, { name: "pipelinetask custom task doesn't support retries", @@ -544,8 +544,8 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { TaskRef: &TaskRef{APIVersion: "example.dev/v0", Kind: "Example"}, }}, expectedError: apis.FieldError{ - Message: `GMD FIX ME`, - Paths: []string{"tasks[0].taskSpec.steps"}, + Message: `invalid value: custom tasks do not support retries`, + Paths: []string{"tasks[0].retries"}, }, }, { name: "pipelinetask custom task doesn't support pipeline resources", @@ -555,8 +555,8 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { TaskRef: &TaskRef{APIVersion: "example.dev/v0", Kind: "Example"}, }}, expectedError: apis.FieldError{ - Message: `GMD FIX ME`, - Paths: []string{"tasks[0].taskSpec.steps"}, + Message: `invalid value: custom tasks do not support PipelineResources`, + Paths: []string{"tasks[0].resources"}, }, }, { name: "pipelinetask custom task doesn't support workspaces", @@ -565,6 +565,10 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { Workspaces: []WorkspacePipelineTaskBinding{{}}, TaskRef: &TaskRef{APIVersion: "example.dev/v0", Kind: "Example"}, }}, + expectedError: apis.FieldError{ + Message: `invalid value: custom tasks do not support Workspaces`, + Paths: []string{"tasks[0].workspaces"}, + }, }, { name: "pipelinetask custom task doesn't support timeout", tasks: []PipelineTask{{ @@ -573,8 +577,8 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { TaskRef: &TaskRef{APIVersion: "example.dev/v0", Kind: "Example"}, }}, expectedError: apis.FieldError{ - Message: `GMD FIX ME`, - Paths: []string{"tasks[0].taskSpec.steps"}, + Message: `invalid value: custom tasks do not support timeout`, + Paths: []string{"tasks[0].timeout"}, }, }} { t.Run(tc.name, func(t *testing.T) { @@ -582,7 +586,7 @@ func TestValidatePipelineTasks_Failure(t *testing.T) { if err == nil { t.Error("Pipeline.validatePipelineTasks() did not return error for invalid pipeline tasks") } - if d := cmp.Diff(tt.expectedError.Error(), err.Error(), cmpopts.IgnoreUnexported(apis.FieldError{})); d != "" { + if d := cmp.Diff(tc.expectedError.Error(), err.Error(), cmpopts.IgnoreUnexported(apis.FieldError{})); d != "" { t.Errorf("PipelineSpec.Validate() errors diff %s", diff.PrintWantGot(d)) } }) diff --git a/pkg/apis/pipeline/v1beta1/pipelinerun_types.go b/pkg/apis/pipeline/v1beta1/pipelinerun_types.go index 79683c69c10..303db537a2a 100644 --- a/pkg/apis/pipeline/v1beta1/pipelinerun_types.go +++ b/pkg/apis/pipeline/v1beta1/pipelinerun_types.go @@ -21,6 +21,7 @@ import ( "github.com/tektoncd/pipeline/pkg/apis/config" "github.com/tektoncd/pipeline/pkg/apis/pipeline" + runv1alpha1 "github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime/schema" @@ -256,6 +257,9 @@ func (pr *PipelineRunStatus) InitializeConditions() { if pr.TaskRuns == nil { pr.TaskRuns = make(map[string]*PipelineRunTaskRunStatus) } + if pr.Runs == nil { + pr.Runs = make(map[string]*PipelineRunRunStatus) + } if pr.StartTime.IsZero() { pr.StartTime = &metav1.Time{Time: time.Now()} started = true @@ -380,8 +384,12 @@ type PipelineRunTaskRunStatus struct { type PipelineRunRunStatus struct { // PipelineTaskName is the name of the PipelineTask. PipelineTaskName string `json:"pipelineTaskName,omitempty"` - - // TODO(#3133): Add v1alpha1.RunStatus here, without introducing an import cycle. + // Status is the RunStatus for the corresponding Run + // +optional + Status *runv1alpha1.RunStatus `json:"status,omitempty"` + // WhenExpressions is the list of checks guarding the execution of the PipelineTask + // +optional + WhenExpressions []WhenExpression `json:"whenExpressions,omitempty"` } // PipelineRunConditionCheckStatus returns the condition check status diff --git a/pkg/apis/pipeline/v1beta1/pipelinerun_types_test.go b/pkg/apis/pipeline/v1beta1/pipelinerun_types_test.go index 4c5785a9fe5..596e15036c1 100644 --- a/pkg/apis/pipeline/v1beta1/pipelinerun_types_test.go +++ b/pkg/apis/pipeline/v1beta1/pipelinerun_types_test.go @@ -91,7 +91,11 @@ func TestInitializePipelineRunConditions(t *testing.T) { p.Status.InitializeConditions() if p.Status.TaskRuns == nil { - t.Fatalf("PipelineRun status not initialized correctly") + t.Fatalf("PipelineRun TaskRun status not initialized correctly") + } + + if p.Status.Runs == nil { + t.Fatalf("PipelineRun Run status not initialized correctly") } if p.Status.StartTime.IsZero() { @@ -103,6 +107,7 @@ func TestInitializePipelineRunConditions(t *testing.T) { t.Fatalf("PipelineRun initialize reason should be %s, got %s instead", v1beta1.PipelineRunReasonStarted.String(), condition.Reason) } p.Status.TaskRuns["fooTask"] = &v1beta1.PipelineRunTaskRunStatus{} + p.Status.Runs["bahTask"] = &v1beta1.PipelineRunRunStatus{} // Change the reason before we initialize again p.Status.SetCondition(&apis.Condition{ @@ -114,7 +119,10 @@ func TestInitializePipelineRunConditions(t *testing.T) { p.Status.InitializeConditions() if len(p.Status.TaskRuns) != 1 { - t.Fatalf("PipelineRun status getting reset") + t.Fatalf("PipelineRun TaskRun status getting reset") + } + if len(p.Status.Runs) != 1 { + t.Fatalf("PipelineRun Run status getting reset") } newCondition := p.Status.GetCondition(apis.ConditionSucceeded) diff --git a/pkg/apis/pipeline/v1beta1/swagger.json b/pkg/apis/pipeline/v1beta1/swagger.json index 46c3df3030e..e2156137759 100644 --- a/pkg/apis/pipeline/v1beta1/swagger.json +++ b/pkg/apis/pipeline/v1beta1/swagger.json @@ -832,6 +832,27 @@ } } }, + "v1beta1.PipelineRunRunStatus": { + "description": "PipelineRunRunStatus contains the name of the PipelineTask for this Run and the Run's Status", + "type": "object", + "properties": { + "pipelineTaskName": { + "description": "PipelineTaskName is the name of the PipelineTask.", + "type": "string" + }, + "status": { + "description": "Status is the RunStatus for the corresponding Run", + "$ref": "#/definitions/github.com.tektoncd.pipeline.pkg.apis.run.v1alpha1.RunStatus" + }, + "whenExpressions": { + "description": "WhenExpressions is the list of checks guarding the execution of the PipelineTask", + "type": "array", + "items": { + "$ref": "#/definitions/v1beta1.WhenExpression" + } + } + } + }, "v1beta1.PipelineRunSpec": { "description": "PipelineRunSpec defines the desired state of PipelineRun", "type": "object", @@ -946,6 +967,13 @@ "description": "PipelineRunSpec contains the exact spec used to instantiate the run", "$ref": "#/definitions/v1beta1.PipelineSpec" }, + "runs": { + "description": "map of PipelineRunRunStatus with the run name as the key", + "type": "object", + "additionalProperties": { + "$ref": "#/definitions/v1beta1.PipelineRunRunStatus" + } + }, "skippedTasks": { "description": "list of tasks that were skipped due to when expressions evaluating to false", "type": "array", @@ -985,6 +1013,13 @@ "description": "PipelineRunSpec contains the exact spec used to instantiate the run", "$ref": "#/definitions/v1beta1.PipelineSpec" }, + "runs": { + "description": "map of PipelineRunRunStatus with the run name as the key", + "type": "object", + "additionalProperties": { + "$ref": "#/definitions/v1beta1.PipelineRunRunStatus" + } + }, "skippedTasks": { "description": "list of tasks that were skipped due to when expressions evaluating to false", "type": "array", diff --git a/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go b/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go index 8e9fbb7178d..2905500acbe 100644 --- a/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go +++ b/pkg/apis/pipeline/v1beta1/zz_generated.deepcopy.go @@ -23,6 +23,7 @@ package v1beta1 import ( pod "github.com/tektoncd/pipeline/pkg/apis/pipeline/pod" v1alpha1 "github.com/tektoncd/pipeline/pkg/apis/resource/v1alpha1" + runv1alpha1 "github.com/tektoncd/pipeline/pkg/apis/run/v1alpha1" v1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" runtime "k8s.io/apimachinery/pkg/runtime" @@ -590,6 +591,18 @@ func (in *PipelineRunResult) DeepCopy() *PipelineRunResult { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *PipelineRunRunStatus) DeepCopyInto(out *PipelineRunRunStatus) { *out = *in + if in.Status != nil { + in, out := &in.Status, &out.Status + *out = new(runv1alpha1.RunStatus) + (*in).DeepCopyInto(*out) + } + if in.WhenExpressions != nil { + in, out := &in.WhenExpressions, &out.WhenExpressions + *out = make([]WhenExpression, len(*in)) + for i := range *in { + (*in)[i].DeepCopyInto(&(*out)[i]) + } + } return } @@ -742,7 +755,7 @@ func (in *PipelineRunStatusFields) DeepCopyInto(out *PipelineRunStatusFields) { } else { in, out := &val, &outVal *out = new(PipelineRunRunStatus) - **out = **in + (*in).DeepCopyInto(*out) } (*out)[key] = outVal } diff --git a/pkg/apis/run/v1alpha1/runstatus_types.go b/pkg/apis/run/v1alpha1/runstatus_types.go new file mode 100644 index 00000000000..caab02e4deb --- /dev/null +++ b/pkg/apis/run/v1alpha1/runstatus_types.go @@ -0,0 +1,144 @@ +/* +Copyright 2020 The Tekton Authors + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +package v1alpha1 + +import ( + "encoding/json" + "time" + + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "k8s.io/apimachinery/pkg/runtime" + "knative.dev/pkg/apis" + duckv1 "knative.dev/pkg/apis/duck/v1" +) + +// This package exists to avoid an import cycle between v1alpha1 and v1beta1. +// It contains common definitions needed by v1alpha1.Run and v1beta1.PipelineRun. + +// +k8s:deepcopy-gen=true + +// RunStatus defines the observed state of Run +type RunStatus struct { + duckv1.Status `json:",inline"` + + // RunStatusFields inlines the status fields. + RunStatusFields `json:",inline"` +} + +// +k8s:deepcopy-gen=true + +// RunStatusFields holds the fields of Run's status. This is defined +// separately and inlined so that other types can readily consume these fields +// via duck typing. +type RunStatusFields struct { + // StartTime is the time the build is actually started. + // +optional + StartTime *metav1.Time `json:"startTime,omitempty"` + + // CompletionTime is the time the build completed. + // +optional + CompletionTime *metav1.Time `json:"completionTime,omitempty"` + + // Results reports any output result values to be consumed by later + // tasks in a pipeline. + // +optional + Results []RunResult `json:"results,omitempty"` + + // ExtraFields holds arbitrary fields provided by the custom task + // controller. + ExtraFields runtime.RawExtension `json:"extraFields,omitempty"` +} + +// RunResult used to describe the results of a task +type RunResult struct { + // Name the given name + Name string `json:"name"` + // Value the given value of the result + Value string `json:"value"` +} + +var runCondSet = apis.NewBatchConditionSet() + +// GetCondition returns the Condition matching the given type. +func (r *RunStatus) GetCondition(t apis.ConditionType) *apis.Condition { + return runCondSet.Manage(r).GetCondition(t) +} + +// InitializeConditions will set all conditions in runCondSet to unknown for the PipelineRun +// and set the started time to the current time +func (r *RunStatus) InitializeConditions() { + started := false + if r.StartTime.IsZero() { + r.StartTime = &metav1.Time{Time: time.Now()} + started = true + } + conditionManager := runCondSet.Manage(r) + conditionManager.InitializeConditions() + // Ensure the started reason is set for the "Succeeded" condition + if started { + initialCondition := conditionManager.GetCondition(apis.ConditionSucceeded) + initialCondition.Reason = "Started" + conditionManager.SetCondition(*initialCondition) + } +} + +// SetCondition sets the condition, unsetting previous conditions with the same +// type as necessary. +func (r *RunStatus) SetCondition(newCond *apis.Condition) { + if newCond != nil { + runCondSet.Manage(r).SetCondition(*newCond) + } +} + +// MarkRunSucceeded changes the Succeeded condition to True with the provided reason and message. +func (r *RunStatus) MarkRunSucceeded(reason, messageFormat string, messageA ...interface{}) { + runCondSet.Manage(r).MarkTrueWithReason(apis.ConditionSucceeded, reason, messageFormat, messageA...) + succeeded := r.GetCondition(apis.ConditionSucceeded) + r.CompletionTime = &succeeded.LastTransitionTime.Inner +} + +// MarkRunFailed changes the Succeeded condition to False with the provided reason and message. +func (r *RunStatus) MarkRunFailed(reason, messageFormat string, messageA ...interface{}) { + runCondSet.Manage(r).MarkFalse(apis.ConditionSucceeded, reason, messageFormat, messageA...) + succeeded := r.GetCondition(apis.ConditionSucceeded) + r.CompletionTime = &succeeded.LastTransitionTime.Inner +} + +// MarkRunRunning changes the Succeeded condition to Unknown with the provided reason and message. +func (r *RunStatus) MarkRunRunning(reason, messageFormat string, messageA ...interface{}) { + runCondSet.Manage(r).MarkUnknown(apis.ConditionSucceeded, reason, messageFormat, messageA...) +} + +// DecodeExtraFields deserializes the extra fields in the Run status. +func (r *RunStatus) DecodeExtraFields(into interface{}) error { + if len(r.ExtraFields.Raw) == 0 { + return nil + } + return json.Unmarshal(r.ExtraFields.Raw, into) +} + +// EncodeExtraFields serializes the extra fields in the Run status. +func (r *RunStatus) EncodeExtraFields(from interface{}) error { + data, err := json.Marshal(from) + if err != nil { + return err + } + r.ExtraFields = runtime.RawExtension{ + Raw: data, + } + return nil +} diff --git a/pkg/apis/run/v1alpha1/zz_generated.deepcopy.go b/pkg/apis/run/v1alpha1/zz_generated.deepcopy.go new file mode 100644 index 00000000000..b4eb0a2c4c6 --- /dev/null +++ b/pkg/apis/run/v1alpha1/zz_generated.deepcopy.go @@ -0,0 +1,69 @@ +// +build !ignore_autogenerated + +/* +Copyright 2020 The Tekton Authors + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +// Code generated by deepcopy-gen. DO NOT EDIT. + +package v1alpha1 + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *RunStatus) DeepCopyInto(out *RunStatus) { + *out = *in + in.Status.DeepCopyInto(&out.Status) + in.RunStatusFields.DeepCopyInto(&out.RunStatusFields) + return +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunStatus. +func (in *RunStatus) DeepCopy() *RunStatus { + if in == nil { + return nil + } + out := new(RunStatus) + in.DeepCopyInto(out) + return out +} + +// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. +func (in *RunStatusFields) DeepCopyInto(out *RunStatusFields) { + *out = *in + if in.StartTime != nil { + in, out := &in.StartTime, &out.StartTime + *out = (*in).DeepCopy() + } + if in.CompletionTime != nil { + in, out := &in.CompletionTime, &out.CompletionTime + *out = (*in).DeepCopy() + } + if in.Results != nil { + in, out := &in.Results, &out.Results + *out = make([]RunResult, len(*in)) + copy(*out, *in) + } + in.ExtraFields.DeepCopyInto(&out.ExtraFields) + return +} + +// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RunStatusFields. +func (in *RunStatusFields) DeepCopy() *RunStatusFields { + if in == nil { + return nil + } + out := new(RunStatusFields) + in.DeepCopyInto(out) + return out +} diff --git a/pkg/reconciler/pipelinerun/cancel.go b/pkg/reconciler/pipelinerun/cancel.go index b7d18651974..7ec969bf618 100644 --- a/pkg/reconciler/pipelinerun/cancel.go +++ b/pkg/reconciler/pipelinerun/cancel.go @@ -24,6 +24,7 @@ import ( "strings" "time" + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" clientset "github.com/tektoncd/pipeline/pkg/client/clientset/versioned" "go.uber.org/zap" @@ -34,17 +35,25 @@ import ( "knative.dev/pkg/apis" ) -var cancelPatchBytes []byte +var cancelTaskRunPatchBytes, cancelRunPatchBytes []byte func init() { var err error - cancelPatchBytes, err = json.Marshal([]jsonpatch.JsonPatchOperation{{ + cancelTaskRunPatchBytes, err = json.Marshal([]jsonpatch.JsonPatchOperation{{ Operation: "add", Path: "/spec/status", Value: v1beta1.TaskRunSpecStatusCancelled, }}) if err != nil { - log.Fatalf("failed to marshal cancel patch bytes: %v", err) + log.Fatalf("failed to marshal TaskRun cancel patch bytes: %v", err) + } + cancelRunPatchBytes, err = json.Marshal([]jsonpatch.JsonPatchOperation{{ + Operation: "add", + Path: "/spec/status", + Value: v1alpha1.RunSpecStatusCancelled, + }}) + if err != nil { + log.Fatalf("failed to marshal Run cancel patch bytes: %v", err) } } @@ -57,11 +66,20 @@ func cancelPipelineRun(ctx context.Context, logger *zap.SugaredLogger, pr *v1bet for taskRunName := range pr.Status.TaskRuns { logger.Infof("cancelling TaskRun %s", taskRunName) - if _, err := clientSet.TektonV1beta1().TaskRuns(pr.Namespace).Patch(ctx, taskRunName, types.JSONPatchType, cancelPatchBytes, metav1.PatchOptions{}, ""); err != nil { + if _, err := clientSet.TektonV1beta1().TaskRuns(pr.Namespace).Patch(ctx, taskRunName, types.JSONPatchType, cancelTaskRunPatchBytes, metav1.PatchOptions{}, ""); err != nil { errs = append(errs, fmt.Errorf("Failed to patch TaskRun `%s` with cancellation: %s", taskRunName, err).Error()) continue } } + // Loop over the Runs in the PipelineRun status. + for runName := range pr.Status.Runs { + logger.Infof("cancelling Run %s", runName) + + if _, err := clientSet.TektonV1alpha1().Runs(pr.Namespace).Patch(ctx, runName, types.JSONPatchType, cancelRunPatchBytes, metav1.PatchOptions{}, ""); err != nil { + errs = append(errs, fmt.Errorf("Failed to patch Run `%s` with cancellation: %s", runName, err).Error()) + continue + } + } // If we successfully cancelled all the TaskRuns, we can consider the PipelineRun cancelled. if len(errs) == 0 { pr.Status.SetCondition(&apis.Condition{ diff --git a/pkg/reconciler/pipelinerun/cancel_test.go b/pkg/reconciler/pipelinerun/cancel_test.go index c5d1d0bf1a1..4f786effd29 100644 --- a/pkg/reconciler/pipelinerun/cancel_test.go +++ b/pkg/reconciler/pipelinerun/cancel_test.go @@ -20,6 +20,7 @@ import ( "context" "testing" + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" ttesting "github.com/tektoncd/pipeline/pkg/reconciler/testing" "github.com/tektoncd/pipeline/test" @@ -33,6 +34,7 @@ func TestCancelPipelineRun(t *testing.T) { name string pipelineRun *v1beta1.PipelineRun taskRuns []*v1beta1.TaskRun + runs []*v1alpha1.Run }{{ name: "no-resolved-taskrun", pipelineRun: &v1beta1.PipelineRun{ @@ -75,6 +77,24 @@ func TestCancelPipelineRun(t *testing.T) { {ObjectMeta: metav1.ObjectMeta{Name: "t1"}}, {ObjectMeta: metav1.ObjectMeta{Name: "t2"}}, }, + }, { + name: "multiple-runs", + pipelineRun: &v1beta1.PipelineRun{ + ObjectMeta: metav1.ObjectMeta{Name: "test-pipeline-run-cancelled"}, + Spec: v1beta1.PipelineRunSpec{ + Status: v1beta1.PipelineRunSpecStatusCancelled, + }, + Status: v1beta1.PipelineRunStatus{PipelineRunStatusFields: v1beta1.PipelineRunStatusFields{ + Runs: map[string]*v1beta1.PipelineRunRunStatus{ + "t1": {PipelineTaskName: "task-1"}, + "t2": {PipelineTaskName: "task-2"}, + }, + }}, + }, + runs: []*v1alpha1.Run{ + {ObjectMeta: metav1.ObjectMeta{Name: "t1"}}, + {ObjectMeta: metav1.ObjectMeta{Name: "t2"}}, + }, }} for _, tc := range testCases { tc := tc @@ -82,6 +102,7 @@ func TestCancelPipelineRun(t *testing.T) { d := test.Data{ PipelineRuns: []*v1beta1.PipelineRun{tc.pipelineRun}, TaskRuns: tc.taskRuns, + Runs: tc.runs, } ctx, _ := ttesting.SetupFakeContext(t) ctx, cancel := context.WithCancel(ctx) @@ -95,13 +116,26 @@ func TestCancelPipelineRun(t *testing.T) { if cond.IsTrue() { t.Errorf("Expected PipelineRun status to be complete and false, but was %v", cond) } - l, err := c.Pipeline.TektonV1beta1().TaskRuns("").List(ctx, metav1.ListOptions{}) - if err != nil { - t.Fatal(err) + if tc.taskRuns != nil { + l, err := c.Pipeline.TektonV1beta1().TaskRuns("").List(ctx, metav1.ListOptions{}) + if err != nil { + t.Fatal(err) + } + for _, tr := range l.Items { + if tr.Spec.Status != v1beta1.TaskRunSpecStatusCancelled { + t.Errorf("expected task %q to be marked as cancelled, was %q", tr.Name, tr.Spec.Status) + } + } } - for _, tr := range l.Items { - if tr.Spec.Status != v1beta1.TaskRunSpecStatusCancelled { - t.Errorf("expected task %q to be marked as cancelled, was %q", tr.Name, tr.Spec.Status) + if tc.runs != nil { + l, err := c.Pipeline.TektonV1alpha1().Runs("").List(ctx, metav1.ListOptions{}) + if err != nil { + t.Fatal(err) + } + for _, r := range l.Items { + if r.Spec.Status != v1alpha1.RunSpecStatusCancelled { + t.Errorf("expected Run %q to be marked as cancelled, was %q", r.Name, r.Spec.Status) + } } } }) diff --git a/pkg/reconciler/pipelinerun/pipelinerun.go b/pkg/reconciler/pipelinerun/pipelinerun.go index c409abd9710..f3dce6f3be4 100644 --- a/pkg/reconciler/pipelinerun/pipelinerun.go +++ b/pkg/reconciler/pipelinerun/pipelinerun.go @@ -187,11 +187,14 @@ func (c *Reconciler) ReconcileKind(ctx context.Context, pr *v1beta1.PipelineRun) return c.finishReconcileUpdateEmitEvents(ctx, pr, before, err) } c.timeoutHandler.Release(pr.GetNamespacedName()) - // TODO(#3133): Also updateRunsStatusDirectly to update status of Custom Task Runs. if err := c.updateTaskRunsStatusDirectly(pr); err != nil { logger.Errorf("Failed to update TaskRun status for PipelineRun %s: %v", pr.Name, err) return c.finishReconcileUpdateEmitEvents(ctx, pr, before, err) } + if err := c.updateRunsStatusDirectly(pr); err != nil { + logger.Errorf("Failed to update Run status for PipelineRun %s: %v", pr.Name, err) + return c.finishReconcileUpdateEmitEvents(ctx, pr, before, err) + } go func(metrics *Recorder) { err := metrics.DurationAndCount(pr) if err != nil { @@ -467,7 +470,6 @@ func (c *Reconciler) reconcile(ctx context.Context, pr *v1beta1.PipelineRun, get tasks = append(tasks, pipelineSpec.Finally...) } pipelineRunState, err := c.resolvePipelineState(ctx, tasks, pipelineMeta, pr, providedResources) - if err != nil { return err } @@ -538,7 +540,7 @@ func (c *Reconciler) reconcile(ctx context.Context, pr *v1beta1.PipelineRun, get after = pr.Status.GetCondition(apis.ConditionSucceeded) pr.Status.StartTime = pipelineRunFacts.State.AdjustStartTime(pr.Status.StartTime) pr.Status.TaskRuns = pipelineRunFacts.State.GetTaskRunsStatus(pr) - pr.Status.Runs = pipelineRunFacts.getRunsStatus(pr) + pr.Status.Runs = pipelineRunFacts.State.GetRunsStatus(pr) pr.Status.SkippedTasks = pipelineRunFacts.GetSkippedTasks() logger.Infof("PipelineRun %s status is being set to %s", pr.Name, after) return nil @@ -548,6 +550,7 @@ func (c *Reconciler) reconcile(ctx context.Context, pr *v1beta1.PipelineRun, get // pipeline run state, and starts them // after all DAG tasks are done, it's responsible for scheduling final tasks and start executing them func (c *Reconciler) runNextSchedulableTask(ctx context.Context, pr *v1beta1.PipelineRun, pipelineRunFacts *resources.PipelineRunFacts, as artifacts.ArtifactStorageInterface) error { + logger := logging.FromContext(ctx) recorder := controller.GetEventRecorder(ctx) @@ -627,11 +630,29 @@ func (c *Reconciler) updateTaskRunsStatusDirectly(pr *v1beta1.PipelineRun) error // TODO(dibyom): Add conditionCheck statuses here prtrs := pr.Status.TaskRuns[taskRunName] tr, err := c.taskRunLister.TaskRuns(pr.Namespace).Get(taskRunName) - if err != nil && !errors.IsNotFound(err) { + if err != nil { // If the TaskRun isn't found, it just means it won't be run - return fmt.Errorf("error retrieving TaskRun %s: %w", taskRunName, err) + if !errors.IsNotFound(err) { + return fmt.Errorf("error retrieving TaskRun %s: %w", taskRunName, err) + } + } else { + prtrs.Status = &tr.Status + } + } + return nil +} + +func (c *Reconciler) updateRunsStatusDirectly(pr *v1beta1.PipelineRun) error { + for runName := range pr.Status.Runs { + prRunStatus := pr.Status.Runs[runName] + run, err := c.runLister.Runs(pr.Namespace).Get(runName) + if err != nil { + if !errors.IsNotFound(err) { + return fmt.Errorf("error retrieving Run %s: %w", runName, err) + } + } else { + prRunStatus.Status = &run.Status } - prtrs.Status = &tr.Status } return nil } @@ -702,6 +723,7 @@ func (c *Reconciler) createTaskRun(ctx context.Context, rprt *resources.Resolved } func (c *Reconciler) createRun(ctx context.Context, rprt *resources.ResolvedPipelineRunTask, pr *v1beta1.PipelineRun) (*v1alpha1.Run, error) { + logger := logging.FromContext(ctx) r := &v1alpha1.Run{ ObjectMeta: metav1.ObjectMeta{ Name: rprt.RunName, @@ -715,7 +737,8 @@ func (c *Reconciler) createRun(ctx context.Context, rprt *resources.ResolvedPipe Params: rprt.PipelineTask.Params, }, } - return c.PipelineClientSet.TektonV1alpha1().Runs(pr.Namespace).Create(r) + logger.Infof("Creating a new Run object %s", rprt.RunName) + return c.PipelineClientSet.TektonV1alpha1().Runs(pr.Namespace).Create(ctx, r, metav1.CreateOptions{}) } // taskWorkspaceByWorkspaceVolumeSource is returning the WorkspaceBinding with the TaskRun specified name. @@ -955,28 +978,42 @@ func (c *Reconciler) updatePipelineRunStatusFromInformer(ctx context.Context, pr logger.Errorf("could not list TaskRuns %#v", err) return err } - pr.Status = updatePipelineRunStatusFromTaskRuns(logger, pr.Name, pr.Status, taskRuns) + updatePipelineRunStatusFromTaskRuns(logger, pr, taskRuns) + + runs, err := c.runLister.Runs(pr.Namespace).List(labels.SelectorFromSet(pipelineRunLabels)) + if err != nil { + logger.Errorf("could not list Runs %#v", err) + return err + } + updatePipelineRunStatusFromRuns(logger, pr, runs) + return nil } -func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, prName string, prStatus v1beta1.PipelineRunStatus, trs []*v1beta1.TaskRun) v1beta1.PipelineRunStatus { +func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, pr *v1beta1.PipelineRun, trs []*v1beta1.TaskRun) { // If no TaskRun was found, nothing to be done. We never remove taskruns from the status if trs == nil || len(trs) == 0 { - return prStatus + return } // Store a list of Condition TaskRuns for each PipelineTask (by name) conditionTaskRuns := make(map[string][]*v1beta1.TaskRun) // Map PipelineTask names to TaskRun names that were already in the status taskRunByPipelineTask := make(map[string]string) - if prStatus.TaskRuns != nil { - for taskRunName, pipelineRunTaskRunStatus := range prStatus.TaskRuns { + if pr.Status.TaskRuns != nil { + for taskRunName, pipelineRunTaskRunStatus := range pr.Status.TaskRuns { taskRunByPipelineTask[pipelineRunTaskRunStatus.PipelineTaskName] = taskRunName } } else { - prStatus.TaskRuns = make(map[string]*v1beta1.PipelineRunTaskRunStatus) + pr.Status.TaskRuns = make(map[string]*v1beta1.PipelineRunTaskRunStatus) } // Loop over all the TaskRuns associated to Tasks for _, taskrun := range trs { + // Only process TaskRuns that are owned by this PipelineRun. + // This skips TaskRuns that are indirectly created by the PipelineRun (e.g. by custom tasks). + if len(taskrun.OwnerReferences) < 1 || taskrun.OwnerReferences[0].UID != pr.ObjectMeta.UID { + logger.Infof("Found a TaskRun %s that is not owned by this PipelineRun", taskrun.Name) + continue + } lbls := taskrun.GetLabels() pipelineTaskName := lbls[pipeline.GroupName+pipeline.PipelineTaskLabelKey] if _, ok := lbls[pipeline.GroupName+pipeline.ConditionCheckKey]; ok { @@ -988,10 +1025,10 @@ func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, prName strin conditionTaskRuns[pipelineTaskName] = append(conditionTaskRuns[pipelineTaskName], taskrun) continue } - if _, ok := prStatus.TaskRuns[taskrun.Name]; !ok { + if _, ok := pr.Status.TaskRuns[taskrun.Name]; !ok { // This taskrun was missing from the status. // Add it without conditions, which are handled in the next loop - prStatus.TaskRuns[taskrun.Name] = &v1beta1.PipelineRunTaskRunStatus{ + pr.Status.TaskRuns[taskrun.Name] = &v1beta1.PipelineRunTaskRunStatus{ PipelineTaskName: pipelineTaskName, Status: &taskrun.Status, ConditionChecks: nil, @@ -1008,8 +1045,8 @@ func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, prName strin // status. This means that the conditions were orphaned, and never added to the // status. In this case we need to generate a new TaskRun name, that will be used // to run the TaskRun if the conditions are passed. - taskRunName = resources.GetTaskRunName(prStatus.TaskRuns, pipelineTaskName, prName) - prStatus.TaskRuns[taskRunName] = &v1beta1.PipelineRunTaskRunStatus{ + taskRunName = resources.GetTaskRunName(pr.Status.TaskRuns, pipelineTaskName, pr.Name) + pr.Status.TaskRuns[taskRunName] = &v1beta1.PipelineRunTaskRunStatus{ PipelineTaskName: pipelineTaskName, Status: nil, ConditionChecks: nil, @@ -1017,7 +1054,7 @@ func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, prName strin } // Build the map of condition checks for the taskrun // If there were no other condition, initialise the map - conditionChecks := prStatus.TaskRuns[taskRunName].ConditionChecks + conditionChecks := pr.Status.TaskRuns[taskRunName].ConditionChecks if conditionChecks == nil { conditionChecks = make(map[string]*v1beta1.PipelineRunConditionCheckStatus) } @@ -1038,7 +1075,34 @@ func updatePipelineRunStatusFromTaskRuns(logger *zap.SugaredLogger, prName strin } } } - prStatus.TaskRuns[taskRunName].ConditionChecks = conditionChecks + pr.Status.TaskRuns[taskRunName].ConditionChecks = conditionChecks + } +} + +func updatePipelineRunStatusFromRuns(logger *zap.SugaredLogger, pr *v1beta1.PipelineRun, runs []*v1alpha1.Run) { + // If no Run was found, nothing to be done. We never remove runs from the status + if runs == nil || len(runs) == 0 { + return + } + if pr.Status.Runs == nil { + pr.Status.Runs = make(map[string]*v1beta1.PipelineRunRunStatus) + } + // Loop over all the Runs associated to Tasks + for _, run := range runs { + // Only process Runs that are owned by this PipelineRun. + // This skips Runs that are indirectly created by the PipelineRun (e.g. by custom tasks). + if len(run.OwnerReferences) < 1 && run.OwnerReferences[0].UID != pr.ObjectMeta.UID { + logger.Infof("Found a Run %s that is not owned by this PipelineRun", run.Name) + continue + } + lbls := run.GetLabels() + pipelineTaskName := lbls[pipeline.GroupName+pipeline.PipelineTaskLabelKey] + if _, ok := pr.Status.Runs[run.Name]; !ok { + // This run was missing from the status. + pr.Status.Runs[run.Name] = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: pipelineTaskName, + Status: &run.Status, + } + } } - return prStatus } diff --git a/pkg/reconciler/pipelinerun/pipelinerun_test.go b/pkg/reconciler/pipelinerun/pipelinerun_test.go index 7aa4e57bad6..0028694fb9a 100644 --- a/pkg/reconciler/pipelinerun/pipelinerun_test.go +++ b/pkg/reconciler/pipelinerun/pipelinerun_test.go @@ -55,6 +55,7 @@ import ( ktesting "k8s.io/client-go/testing" "k8s.io/client-go/tools/record" "knative.dev/pkg/apis" + duckv1 "knative.dev/pkg/apis/duck/v1" duckv1beta1 "knative.dev/pkg/apis/duck/v1beta1" "knative.dev/pkg/configmap" "knative.dev/pkg/controller" @@ -471,6 +472,10 @@ func TestReconcile_CustomTask(t *testing.T) { PipelineSpec: &v1beta1.PipelineSpec{ Tasks: []v1beta1.PipelineTask{{ Name: "custom-task", + Params: []v1beta1.Param{{ + Name: "param1", + Value: v1beta1.ArrayOrString{Type: v1beta1.ParamTypeString, StringVal: "value1"}, + }}, TaskRef: &v1beta1.TaskRef{ APIVersion: "example.dev/v0", Kind: "Example", @@ -515,6 +520,10 @@ func TestReconcile_CustomTask(t *testing.T) { Annotations: map[string]string{}, }, Spec: v1alpha1.RunSpec{ + Params: []v1beta1.Param{{ + Name: "param1", + Value: v1beta1.ArrayOrString{Type: v1beta1.ParamTypeString, StringVal: "value1"}, + }}, Ref: &v1beta1.TaskRef{ APIVersion: "example.dev/v0", Kind: "Example", @@ -957,6 +966,78 @@ func TestUpdateTaskRunsState(t *testing.T) { } +func TestUpdateRunsState(t *testing.T) { + // TestUpdateRunsState runs "getRunsStatus" and verifies how it updates a PipelineRun status + // from a Run associated to the PipelineRun + pr := tb.PipelineRun("test-pipeline-run", tb.PipelineRunNamespace("foo"), tb.PipelineRunSpec("test-pipeline")) + pipelineTask := v1beta1.PipelineTask{ + Name: "unit-test-1", + WhenExpressions: []v1beta1.WhenExpression{{ + Input: "foo", + Operator: selection.In, + Values: []string{"foo", "bar"}, + }}, + TaskRef: &v1beta1.TaskRef{ + APIVersion: "example.dev/v0", + Kind: "Example", + Name: "unit-test-run", + }, + } + run := v1alpha1.Run{ + TypeMeta: metav1.TypeMeta{ + APIVersion: "example.dev/v0", + Kind: "Example", + }, + ObjectMeta: metav1.ObjectMeta{ + Name: "unit-test-run", + Namespace: "foo", + }, + Spec: v1alpha1.RunSpec{}, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: duckv1.Conditions{{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionTrue, + }}, + }, + }, + } + expectedRunsStatus := make(map[string]*v1beta1.PipelineRunRunStatus) + expectedRunsStatus["test-pipeline-run-success-unit-test-1"] = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: "unit-test-1", + Status: &v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: duckv1.Conditions{{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionTrue, + }}, + }, + }, + WhenExpressions: []v1beta1.WhenExpression{{ + Input: "foo", + Operator: selection.In, + Values: []string{"foo", "bar"}, + }}, + } + expectedPipelineRunStatus := v1beta1.PipelineRunStatus{ + PipelineRunStatusFields: v1beta1.PipelineRunStatusFields{ + Runs: expectedRunsStatus, + }, + } + + state := resources.PipelineRunState{{ + PipelineTask: &pipelineTask, + RunName: "test-pipeline-run-success-unit-test-1", + Run: &run, + }} + pr.Status.InitializeConditions() + status := state.GetRunsStatus(pr) + if d := cmp.Diff(expectedPipelineRunStatus.Runs, status); d != "" { + t.Fatalf("Expected PipelineRun status to match Run(s) status, but got a mismatch: %s", diff.PrintWantGot(d)) + } + +} + func TestUpdateTaskRunStateWithConditionChecks(t *testing.T) { // TestUpdateTaskRunsState runs "getTaskRunsStatus" and verifies how it updates a PipelineRun status // from several different TaskRun with Conditions associated to the PipelineRun @@ -2960,7 +3041,12 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { tb.PipelineTask("hello-world-1", helloWorldTask.Name), tb.PipelineTask("hello-world-2", helloWorldTask.Name), tb.PipelineTask("hello-world-3", helloWorldTask.Name, tb.PipelineTaskCondition("always-true")), - tb.PipelineTask("hello-world-4", helloWorldTask.Name, tb.PipelineTaskCondition("always-true")))) + tb.PipelineTask("hello-world-4", helloWorldTask.Name, tb.PipelineTaskCondition("always-true")), + tb.PipelineTask("hello-world-5", helloWorldTask.Name), // custom task (corrected below) + )) + + // Update the fifth pipeline task to be a custom task (builder does not support this case). + testPipeline.Spec.Tasks[4].TaskRef = &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example"} // This taskrun is in the pipelinerun status. It completed successfully. taskRunDone := tb.TaskRun("test-pipeline-run-out-of-sync-hello-world-1", @@ -3049,6 +3135,39 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { ), ) + orphanedRun := &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{ + Name: "test-pipeline-run-out-of-sync-hello-world-5", + Namespace: "foo", + OwnerReferences: []metav1.OwnerReference{{ + Kind: "PipelineRun", + Name: prOutOfSyncName, + }}, + Labels: map[string]string{ + pipeline.GroupName + pipeline.PipelineLabelKey: testPipeline.Name, + pipeline.GroupName + pipeline.PipelineRunLabelKey: prOutOfSyncName, + pipeline.GroupName + pipeline.PipelineTaskLabelKey: "hello-world-5", + }, + Annotations: map[string]string{}, + }, + Spec: v1alpha1.RunSpec{ + Ref: &v1beta1.TaskRef{ + APIVersion: "example.dev/v0", + Kind: "Example", + }, + }, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{ + { + Type: apis.ConditionSucceeded, + Status: corev1.ConditionUnknown, + }, + }, + }, + }, + } + prOutOfSync := tb.PipelineRun(prOutOfSyncName, tb.PipelineRunNamespace("foo"), tb.PipelineRunSpec(testPipeline.Name, tb.PipelineRunServiceAccountName("test-sa")), @@ -3074,6 +3193,7 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { ts := []*v1beta1.Task{helloWorldTask} trs := []*v1beta1.TaskRun{taskRunDone, taskRunOrphaned, taskRunWithCondition, taskRunForOrphanedCondition, taskRunForConditionOfOrphanedTaskRun} + runs := []*v1alpha1.Run{orphanedRun} cs := []*v1alpha1.Condition{ tbv1alpha1.Condition("always-true", tbv1alpha1.ConditionNamespace("foo"), tbv1alpha1.ConditionSpec( tbv1alpha1.ConditionSpecCheck("", "foo", tbv1alpha1.Args("bar")), @@ -3086,6 +3206,7 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { Tasks: ts, TaskRuns: trs, Conditions: cs, + Runs: runs, } prt := NewPipelineRunTest(d, t) defer prt.Cancel() @@ -3104,6 +3225,8 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { switch { case action.Matches("create", "taskruns"): t.Errorf("Expected client to not have created a TaskRun, but it did") + case action.Matches("create", "runs"): + t.Errorf("Expected client to not have created a Run, but it did") case action.Matches("update", "pipelineruns"): pipelineUpdates++ case action.Matches("patch", "pipelineruns"): @@ -3128,6 +3251,7 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { } expectedTaskRunsStatus := make(map[string]*v1beta1.PipelineRunTaskRunStatus) + expectedRunsStatus := make(map[string]*v1beta1.PipelineRunRunStatus) // taskRunDone did not change expectedTaskRunsStatus[taskRunDone.Name] = &v1beta1.PipelineRunTaskRunStatus{ PipelineTaskName: "hello-world-1", @@ -3176,6 +3300,20 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { PipelineTaskName: "hello-world-4", ConditionChecks: prccs4, } + // orphanedRun was recovered into the status + expectedRunsStatus[orphanedRun.Name] = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: "hello-world-5", + Status: &v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{ + { + Type: apis.ConditionSucceeded, + Status: corev1.ConditionUnknown, + }, + }, + }, + }, + } // We cannot just diff status directly because the taskrun name for the orphaned condition // is dynamically generated, but we can change the name to allow us to then diff. @@ -3189,10 +3327,16 @@ func TestReconcileOutOfSyncPipelineRun(t *testing.T) { if d := cmp.Diff(reconciledRun.Status.TaskRuns, expectedTaskRunsStatus); d != "" { t.Fatalf("Expected PipelineRun status to match TaskRun(s) status, but got a mismatch: %s", d) } + if d := cmp.Diff(reconciledRun.Status.Runs, expectedRunsStatus); d != "" { + t.Fatalf("Expected PipelineRun status to match Run(s) status, but got a mismatch: %s", d) + } } func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { + prUID := types.UID("9c04a4d9-33b8-4641-ad11-2aceed91af7b") + otherPrUID := types.UID("9c04a4d9-dead-beef-ad11-2aceed91af7b") + // PipelineRunConditionCheckStatus recovered by updatePipelineRunStatusFromTaskRuns // It does not include the status, which is then retrieved via the regular reconcile prccs2Recovered := map[string]*v1beta1.PipelineRunConditionCheckStatus{ @@ -3300,7 +3444,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { prStatusWithEmptyTaskRuns := v1beta1.PipelineRunStatus{ Status: prRunningStatus, PipelineRunStatusFields: v1beta1.PipelineRunStatusFields{ - TaskRuns: nil, + TaskRuns: map[string]*v1beta1.PipelineRunTaskRunStatus{}, }, } @@ -3366,6 +3510,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { Labels: map[string]string{ pipeline.GroupName + pipeline.PipelineTaskLabelKey: "task-1", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, { @@ -3376,6 +3521,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { pipeline.GroupName + pipeline.ConditionCheckKey: "pr-task-2-running-condition-check-xxyyy", pipeline.GroupName + pipeline.ConditionNameKey: "running-condition", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, { @@ -3384,6 +3530,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { Labels: map[string]string{ pipeline.GroupName + pipeline.PipelineTaskLabelKey: "task-3", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, { @@ -3394,6 +3541,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { pipeline.GroupName + pipeline.ConditionCheckKey: "pr-task-3-successful-condition-check-xxyyy", pipeline.GroupName + pipeline.ConditionNameKey: "successful-condition", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, { @@ -3404,6 +3552,19 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { pipeline.GroupName + pipeline.ConditionCheckKey: "pr-task-4-failed-condition-check-xxyyy", pipeline.GroupName + pipeline.ConditionNameKey: "failed-condition", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, + }, + }, + } + + taskRunsFromAnotherPR := []*v1beta1.TaskRun{ + { + ObjectMeta: metav1.ObjectMeta{ + Name: "pr-task-1-xxyyy", + Labels: map[string]string{ + pipeline.GroupName + pipeline.PipelineTaskLabelKey: "task-1", + }, + OwnerReferences: []metav1.OwnerReference{{UID: otherPrUID}}, }, }, } @@ -3434,6 +3595,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { Labels: map[string]string{ pipeline.GroupName + pipeline.PipelineTaskLabelKey: "task-1", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, }, @@ -3448,6 +3610,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { Labels: map[string]string{ pipeline.GroupName + pipeline.PipelineTaskLabelKey: "task-3", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, { @@ -3458,6 +3621,7 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { pipeline.GroupName + pipeline.ConditionCheckKey: "pr-task-3-successful-condition-check-xxyyy", pipeline.GroupName + pipeline.ConditionNameKey: "successful-condition", }, + OwnerReferences: []metav1.OwnerReference{{UID: prUID}}, }, }, }, @@ -3472,6 +3636,11 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { prStatus: prStatusWithOrphans, trs: allTaskRuns, expectedPrStatus: prStatusRecovered, + }, { + prName: "tr-from-another-pr", + prStatus: prStatusWithEmptyTaskRuns, + trs: taskRunsFromAnotherPR, + expectedPrStatus: prStatusWithEmptyTaskRuns, }, } @@ -3480,7 +3649,12 @@ func TestUpdatePipelineRunStatusFromTaskRuns(t *testing.T) { observer, _ := observer.New(zap.InfoLevel) logger := zap.New(observer).Sugar() - actualPrStatus := updatePipelineRunStatusFromTaskRuns(logger, tc.prName, tc.prStatus, tc.trs) + pr := &v1beta1.PipelineRun{ + ObjectMeta: metav1.ObjectMeta{Name: tc.prName, UID: prUID}, + Status: tc.prStatus, + } + updatePipelineRunStatusFromTaskRuns(logger, pr, tc.trs) + actualPrStatus := pr.Status // The TaskRun keys for recovered taskruns will contain a new random key, appended to the // base name that we expect. Replace the random part so we can diff the whole structure diff --git a/pkg/reconciler/pipelinerun/resources/pipelinerunresolution.go b/pkg/reconciler/pipelinerun/resources/pipelinerunresolution.go index 07a831a7b37..7f7ea45a4ff 100644 --- a/pkg/reconciler/pipelinerun/resources/pipelinerunresolution.go +++ b/pkg/reconciler/pipelinerun/resources/pipelinerunresolution.go @@ -28,10 +28,7 @@ import ( "github.com/tektoncd/pipeline/pkg/list" "github.com/tektoncd/pipeline/pkg/names" "github.com/tektoncd/pipeline/pkg/reconciler/taskrun/resources" - "go.uber.org/zap" - corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/errors" - "k8s.io/apimachinery/pkg/util/sets" "knative.dev/pkg/apis" ) @@ -66,11 +63,9 @@ func (e *ConditionNotFoundError) Error() string { type ResolvedPipelineRunTask struct { TaskRunName string TaskRun *v1beta1.TaskRun - // If the PipelineTask is a Custom Task, RunName and Run will be set. - RunName string - Run *v1alpha1.Run - + RunName string + Run *v1alpha1.Run PipelineTask *v1beta1.PipelineTask ResolvedTaskResources *resources.ResolvedTaskResources // ConditionChecks ~~TaskRuns but for evaling conditions @@ -88,23 +83,6 @@ func (t ResolvedPipelineRunTask) IsCustomTask() bool { t.PipelineTask.TaskRef.APIVersion != "" } -func (t ResolvedPipelineRunTask) IsDone() bool { - if t.PipelineTask == nil { - return false - } - - if !t.IsCustomTask() { - if t.TaskRun == nil { - return false - } - status := t.TaskRun.Status.GetCondition(apis.ConditionSucceeded) - retriesDone := len(t.TaskRun.Status.RetriesStatus) - retries := t.PipelineTask.Retries - return status.IsTrue() || status.IsFalse() && retriesDone >= retries - } - return t.Run != nil && t.Run.IsDone() -} - // IsSuccessful returns true only if the taskrun itself has completed successfully func (t ResolvedPipelineRunTask) IsSuccessful() bool { if !t.IsCustomTask() { @@ -144,8 +122,16 @@ func (t ResolvedPipelineRunTask) IsCancelled() bool { return c.IsFalse() && c.Reason == v1beta1.TaskRunReasonCancelled.String() } - // TODO(#3133): Support cancellation of Custom Task runs. - return false + if t.Run == nil { + return false + } + + c := t.Run.Status.GetCondition(apis.ConditionSucceeded) + if c == nil { + return false + } + + return c.IsFalse() && c.Reason == v1alpha1.RunReasonCancelled } // IsStarted returns true only if the PipelineRunTask itself has a TaskRun or @@ -316,7 +302,6 @@ func ValidateServiceaccountMapping(p *v1beta1.PipelineSpec, pr *v1beta1.Pipeline return nil } -<<<<<<< HEAD // ResolvePipelineRunTask retrieves a single Task's instance using the getTask to fetch // the spec. If it is unable to retrieve an instance of a referenced Task, it will return // an error, otherwise it returns a list of all of the Tasks retrieved. It will retrieve diff --git a/pkg/reconciler/pipelinerun/resources/pipelinerunresolution_test.go b/pkg/reconciler/pipelinerun/resources/pipelinerunresolution_test.go index e6b93e53afd..5b17db58b39 100644 --- a/pkg/reconciler/pipelinerun/resources/pipelinerunresolution_test.go +++ b/pkg/reconciler/pipelinerun/resources/pipelinerunresolution_test.go @@ -18,6 +18,7 @@ package resources import ( "context" + "errors" "fmt" "testing" @@ -41,16 +42,15 @@ import ( duckv1beta1 "knative.dev/pkg/apis/duck/v1beta1" ) -func nopGetRun(string) (*v1alpha1.Run, error) { return nil, errors.New("GetRun should not be called") } -func nopGetTask(string) (v1beta1.TaskInterface, error) { +func nopGetRun(string) (*v1alpha1.Run, error) { + return nil, errors.New("GetRun should not be called") +} +func nopGetTask(context.Context, string) (v1beta1.TaskInterface, error) { return nil, errors.New("GetTask should not be called") } func nopGetTaskRun(string) (*v1beta1.TaskRun, error) { return nil, errors.New("GetTaskRun should not be called") } -func nopGetClusterTask(string) (v1beta1.TaskInterface, error) { - return nil, errors.New("GetClusterTask should not be called") -} var pts = []v1beta1.PipelineTask{{ Name: "mytask1", @@ -113,7 +113,10 @@ var pts = []v1beta1.PipelineTask{{ }}, RunAfter: []string{"mytask1"}, }, { - Name: "mytask10-Jason FIXME", + Name: "mytask13", + TaskRef: &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example", Name: "customtask"}, +}, { + Name: "mytask14", TaskRef: &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example", Name: "customtask"}, }} @@ -152,6 +155,20 @@ var trs = []v1beta1.TaskRun{{ Spec: v1beta1.TaskRunSpec{}, }} +var runs = []v1alpha1.Run{{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: "namespace", + Name: "pipelinerun-mytask13", + }, + Spec: v1alpha1.RunSpec{}, +}, { + ObjectMeta: metav1.ObjectMeta{ + Namespace: "namespace", + Name: "pipelinerun-mytask14", + }, + Spec: v1alpha1.RunSpec{}, +}} + var condition = v1alpha1.Condition{ ObjectMeta: metav1.ObjectMeta{ Name: "always-true", @@ -175,23 +192,47 @@ func makeStarted(tr v1beta1.TaskRun) *v1beta1.TaskRun { return newTr } +func makeRunStarted(run v1alpha1.Run) *v1alpha1.Run { + newRun := newRun(run) + newRun.Status.Conditions[0].Status = corev1.ConditionUnknown + return newRun +} + func makeSucceeded(tr v1beta1.TaskRun) *v1beta1.TaskRun { newTr := newTaskRun(tr) newTr.Status.Conditions[0].Status = corev1.ConditionTrue return newTr } +func makeRunSucceeded(run v1alpha1.Run) *v1alpha1.Run { + newRun := newRun(run) + newRun.Status.Conditions[0].Status = corev1.ConditionTrue + return newRun +} + func makeFailed(tr v1beta1.TaskRun) *v1beta1.TaskRun { newTr := newTaskRun(tr) newTr.Status.Conditions[0].Status = corev1.ConditionFalse return newTr } +func makeRunFailed(run v1alpha1.Run) *v1alpha1.Run { + newRun := newRun(run) + newRun.Status.Conditions[0].Status = corev1.ConditionFalse + return newRun +} + func withCancelled(tr *v1beta1.TaskRun) *v1beta1.TaskRun { tr.Status.Conditions[0].Reason = v1beta1.TaskRunSpecStatusCancelled return tr } +func withRunCancelled(run v1alpha1.Run) *v1alpha1.Run { + newRun := newRun(run) + newRun.Status.Conditions[0].Reason = v1alpha1.RunReasonCancelled + return newRun +} + func withCancelledBySpec(tr *v1beta1.TaskRun) *v1beta1.TaskRun { tr.Spec.Status = v1beta1.TaskRunSpecStatusCancelled return tr @@ -236,6 +277,21 @@ func newTaskRun(tr v1beta1.TaskRun) *v1beta1.TaskRun { } } +func newRun(run v1alpha1.Run) *v1alpha1.Run { + return &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{ + Namespace: run.Namespace, + Name: run.Name, + }, + Spec: run.Spec, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{{Type: apis.ConditionSucceeded}}, + }, + }, + } +} + var noneStartedState = PipelineRunState{{ PipelineTask: &pts[0], TaskRunName: "pipelinerun-mytask1", @@ -316,6 +372,46 @@ var allFinishedState = PipelineRunState{{ }, }} +var noRunStartedState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: nil, +}, { + PipelineTask: &pts[13], + RunName: "pipelinerun-mytask14", + Run: nil, +}} + +var oneRunStartedState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunStarted(runs[0]), +}, { + PipelineTask: &pts[13], + RunName: "pipelinerun-mytask14", + Run: nil, +}} + +var oneRunFinishedState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunSucceeded(runs[0]), +}, { + PipelineTask: &pts[13], + RunName: "pipelinerun-mytask14", + Run: nil, +}} + +var oneRunFailedState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunFailed(runs[0]), +}, { + PipelineTask: &pts[13], + RunName: "pipelinerun-mytask14", + Run: nil, +}} + var successTaskConditionCheckState = TaskConditionCheckState{{ ConditionCheckName: "myconditionCheck", Condition: &condition, @@ -485,6 +581,12 @@ var taskCancelled = PipelineRunState{{ }, }} +var runCancelled = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: withRunCancelled(runs[0]), +}} + var taskWithOptionalResourcesDeprecated = &v1beta1.Task{ ObjectMeta: metav1.ObjectMeta{ Name: "task", @@ -855,20 +957,20 @@ func TestIsSkipped(t *testing.T) { "mytask12": false, }, }, { - name: "run-started", - state: oneRunStartedState, + name: "run-started", + state: oneRunStartedState, expected: map[string]bool{ "mytask13": false, }, }, { - name: "run-cancelled", - state: runCancelled, + name: "run-cancelled", + state: runCancelled, expected: map[string]bool{ "mytask13": false, }, }} { t.Run(tc.name, func(t *testing.T) { - dag, err := dagFromState(tc.state) + d, err := dagFromState(tc.state) if err != nil { t.Fatalf("Could not get a dag from the TC state %#v: %v", tc.state, err) } @@ -1098,9 +1200,13 @@ func TestResolvePipelineRun_CustomTask(t *testing.T) { return nil, kerrors.NewNotFound(v1beta1.Resource("run"), name) } nopGetCondition := func(string) (*v1alpha1.Condition, error) { return nil, errors.New("GetCondition should not be called") } - pipelineState, err := ResolvePipelineRun(context.Background(), pr, nopGetTask, nopGetTaskRun, getRun, nopGetClusterTask, nopGetCondition, pts, nil) - if err != nil { - t.Fatalf("ResolvePipelineRun: %v", err) + pipelineState := PipelineRunState{} + for _, task := range pts { + ps, err := ResolvePipelineRunTask(context.Background(), pr, nopGetTask, nopGetTaskRun, getRun, nopGetCondition, task, nil) + if err != nil { + t.Fatalf("ResolvePipelineRunTask: %v", err) + } + pipelineState = append(pipelineState, ps) } expectedState := PipelineRunState{{ @@ -1113,7 +1219,7 @@ func TestResolvePipelineRun_CustomTask(t *testing.T) { Run: run, }} if d := cmp.Diff(expectedState, pipelineState); d != "" { - t.Errorf("Unexpected pipelien state: %s", diff.PrintWantGot(d)) + t.Errorf("Unexpected pipeline state: %s", diff.PrintWantGot(d)) } } @@ -1522,18 +1628,14 @@ func TestResolveConditionChecks_MultipleConditions(t *testing.T) { ResolvedResources: providedResources, }} - for _, tc := range tcs { - t.Run(tc.name, func(t *testing.T) { - ps, err := ResolvePipelineRunTask(context.Background(), pr, getTask, tc.getTaskRun, nopGetRun, getCondition, pt, providedResources) - if err != nil { - t.Fatalf("Did not expect error when resolving PipelineRun without Conditions: %v", err) - } - pipelineState := PipelineRunState{ps} + ps, err := ResolvePipelineRunTask(context.Background(), pr, getTask, getTaskRun, nopGetRun, getCondition, pt, providedResources) + if err != nil { + t.Fatalf("Did not expect error when resolving PipelineRun without Conditions: %v", err) + } + pipelineState := PipelineRunState{ps} - if d := cmp.Diff(expectedConditionCheck, pipelineState[0].ResolvedConditionChecks, cmpopts.IgnoreUnexported(v1beta1.TaskRunSpec{}, ResolvedConditionCheck{})); d != "" { - t.Fatalf("ConditionChecks did not resolve as expected: %s", diff.PrintWantGot(d)) - } - }) + if d := cmp.Diff(expectedConditionCheck, pipelineState[0].ResolvedConditionChecks, cmpopts.IgnoreUnexported(v1beta1.TaskRunSpec{}, ResolvedConditionCheck{})); d != "" { + t.Fatalf("ConditionChecks did not resolve as expected: %s", diff.PrintWantGot(d)) } } func TestResolveConditionChecks_ConditionDoesNotExist(t *testing.T) { @@ -1903,7 +2005,7 @@ func TestResolvePipeline_WhenExpressions(t *testing.T) { } t.Run("When Expressions exist", func(t *testing.T) { - _, err := ResolvePipelineRunTask(context.Background(), pr, getTask, getTaskRun, getCondition, pt, providedResources) + _, err := ResolvePipelineRunTask(context.Background(), pr, getTask, getTaskRun, nopGetRun, getCondition, pt, providedResources) if err != nil { t.Fatalf("Did not expect error when resolving PipelineRun: %v", err) } diff --git a/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go b/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go index f9a5e965d70..0a916b7381a 100644 --- a/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go +++ b/pkg/reconciler/pipelinerun/resources/pipelinerunstate.go @@ -65,7 +65,9 @@ func (state PipelineRunState) ToMap() map[string]*ResolvedPipelineRunTask { // IsBeforeFirstTaskRun returns true if the PipelineRun has not yet started its first TaskRun func (state PipelineRunState) IsBeforeFirstTaskRun() bool { for _, t := range state { - if t.TaskRun != nil { + if t.IsCustomTask() && t.Run != nil { + return false + } else if t.TaskRun != nil { return false } } @@ -84,10 +86,15 @@ func (state PipelineRunState) AdjustStartTime(unadjustedStartTime *metav1.Time) adjustedStartTime := unadjustedStartTime for _, rprt := range state { if rprt.TaskRun == nil { - continue - } - if rprt.TaskRun.CreationTimestamp.Time.Before(adjustedStartTime.Time) { - adjustedStartTime = &rprt.TaskRun.CreationTimestamp + if rprt.Run != nil { + if rprt.Run.CreationTimestamp.Time.Before(adjustedStartTime.Time) { + adjustedStartTime = &rprt.Run.CreationTimestamp + } + } + } else { + if rprt.TaskRun.CreationTimestamp.Time.Before(adjustedStartTime.Time) { + adjustedStartTime = &rprt.TaskRun.CreationTimestamp + } } } return adjustedStartTime.DeepCopy() @@ -99,6 +106,9 @@ func (state PipelineRunState) AdjustStartTime(unadjustedStartTime *metav1.Time) func (state PipelineRunState) GetTaskRunsStatus(pr *v1beta1.PipelineRun) map[string]*v1beta1.PipelineRunTaskRunStatus { status := make(map[string]*v1beta1.PipelineRunTaskRunStatus) for _, rprt := range state { + if rprt.IsCustomTask() { + continue + } if rprt.TaskRun == nil && rprt.ResolvedConditionChecks == nil { continue } @@ -146,21 +156,57 @@ func (state PipelineRunState) GetTaskRunsStatus(pr *v1beta1.PipelineRun) map[str return status } +// GetRunsStatus returns a map of run name and the run. +// Ignore a nil run in pipelineRunState, otherwise, capture run object from PipelineRun Status. +// Update run status based on the pipelineRunState before returning it in the map. +func (state PipelineRunState) GetRunsStatus(pr *v1beta1.PipelineRun) map[string]*v1beta1.PipelineRunRunStatus { + status := map[string]*v1beta1.PipelineRunRunStatus{} + for _, rprt := range state { + if !rprt.IsCustomTask() { + continue + } + if rprt.Run == nil && rprt.ResolvedConditionChecks == nil { + continue + } + + var prrs *v1beta1.PipelineRunRunStatus + if rprt.Run != nil { + prrs = pr.Status.Runs[rprt.RunName] + } + + if prrs == nil { + prrs = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: rprt.PipelineTask.Name, + WhenExpressions: rprt.PipelineTask.WhenExpressions, + } + } + + if rprt.Run != nil { + prrs.Status = &rprt.Run.Status + } + + // TODO(#3133): Include any condition check statuses here too. + status[rprt.RunName] = prrs + } + return status +} + // getNextTasks returns a list of tasks which should be executed next i.e. // a list of tasks from candidateTasks which aren't yet indicated in state to be running and // a list of cancelled/failed tasks from candidateTasks which haven't exhausted their retries func (state PipelineRunState) getNextTasks(candidateTasks sets.String) []*ResolvedPipelineRunTask { tasks := []*ResolvedPipelineRunTask{} for _, t := range state { - if _, ok := candidateTasks[t.PipelineTask.Name]; ok && t.TaskRun == nil { - tasks = append(tasks, t) - } - if _, ok := candidateTasks[t.PipelineTask.Name]; ok && t.TaskRun != nil { - status := t.TaskRun.Status.GetCondition(apis.ConditionSucceeded) - if status != nil && status.IsFalse() { - if !(t.TaskRun.IsCancelled() || status.Reason == v1beta1.TaskRunReasonCancelled.String() || status.Reason == ReasonConditionCheckFailed) { - if len(t.TaskRun.Status.RetriesStatus) < t.PipelineTask.Retries { - tasks = append(tasks, t) + if _, ok := candidateTasks[t.PipelineTask.Name]; ok { + if t.TaskRun == nil && t.Run == nil { + tasks = append(tasks, t) + } else if t.TaskRun != nil { // only TaskRun currently supports retry + status := t.TaskRun.Status.GetCondition(apis.ConditionSucceeded) + if status != nil && status.IsFalse() { + if !(t.TaskRun.IsCancelled() || status.Reason == v1beta1.TaskRunReasonCancelled.String() || status.Reason == ReasonConditionCheckFailed) { + if len(t.TaskRun.Status.RetriesStatus) < t.PipelineTask.Retries { + tasks = append(tasks, t) + } } } } @@ -360,12 +406,12 @@ func (facts *PipelineRunFacts) getPipelineTasksCount() pipelineRunStatusCount { // increment success counter since the task is successful case t.IsSuccessful(): s.Succeeded++ - // increment failure counter since the task has failed - case t.IsFailure(): - s.Failed++ // increment cancelled counter since the task is cancelled case t.IsCancelled(): s.Cancelled++ + // increment failure counter since the task has failed + case t.IsFailure(): + s.Failed++ // increment skip counter since the task is skipped case t.Skip(facts): s.Skipped++ diff --git a/pkg/reconciler/pipelinerun/resources/pipelinerunstate_test.go b/pkg/reconciler/pipelinerun/resources/pipelinerunstate_test.go index 0407c89f19e..1b0946a2f78 100644 --- a/pkg/reconciler/pipelinerun/resources/pipelinerunstate_test.go +++ b/pkg/reconciler/pipelinerun/resources/pipelinerunstate_test.go @@ -22,6 +22,7 @@ import ( "github.com/google/go-cmp/cmp" tb "github.com/tektoncd/pipeline/internal/builder/v1beta1" + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" "github.com/tektoncd/pipeline/pkg/reconciler/pipeline/dag" "github.com/tektoncd/pipeline/pkg/reconciler/taskrun/resources" @@ -31,6 +32,7 @@ import ( metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/util/sets" "knative.dev/pkg/apis" + duckv1 "knative.dev/pkg/apis/duck/v1" duckv1beta1 "knative.dev/pkg/apis/duck/v1beta1" ) @@ -98,6 +100,24 @@ func TestPipelineRunFacts_CheckDAGTasksDoneDone(t *testing.T) { }, }} + var runRunningState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunStarted(runs[0]), + }} + + var runSucceededState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunSucceeded(runs[0]), + }} + + var runFailedState = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: makeRunFailed(runs[0]), + }} + tcs := []struct { name string state PipelineRunState @@ -138,11 +158,26 @@ func TestPipelineRunFacts_CheckDAGTasksDoneDone(t *testing.T) { state: noTaskRunState, expected: false, ptExpected: []bool{false}, + }, { + name: "run-running-no-candidates", + state: runRunningState, + expected: false, + ptExpected: []bool{false}, + }, { + name: "run-succeeded-no-candidates", + state: runSucceededState, + expected: true, + ptExpected: []bool{true}, + }, { + name: "run-failed-no-candidates", + state: runFailedState, + expected: true, + ptExpected: []bool{true}, }} for _, tc := range tcs { t.Run(tc.name, func(t *testing.T) { - d, err := DagFromState(tc.state) + d, err := dagFromState(tc.state) if err != nil { t.Fatalf("Unexpected error while buildig DAG for state %v: %v", tc.state, err) } @@ -173,12 +208,24 @@ func TestIsBeforeFirstTaskRun_WithNotStartedTask(t *testing.T) { } } +func TestIsBeforeFirstTaskRun_WithNotStartedRun(t *testing.T) { + if !noRunStartedState.IsBeforeFirstTaskRun() { + t.Fatalf("Expected state to be before first taskrun (Run test)") + } +} + func TestIsBeforeFirstTaskRun_WithStartedTask(t *testing.T) { if oneStartedState.IsBeforeFirstTaskRun() { t.Fatalf("Expected state to be after first taskrun") } } +func TestIsBeforeFirstTaskRun_WithStartedRun(t *testing.T) { + if oneRunStartedState.IsBeforeFirstTaskRun() { + t.Fatalf("Expected state to be after first taskrun (Run test)") + } +} + func TestGetNextTasks(t *testing.T) { tcs := []struct { name string @@ -290,6 +337,21 @@ func TestGetNextTasks(t *testing.T) { state: taskCancelled, candidates: sets.NewString("mytask5"), expectedNext: []*ResolvedPipelineRunTask{}, + }, { + name: "no-runs-started-both-candidates", + state: noRunStartedState, + candidates: sets.NewString("mytask13", "mytask14"), + expectedNext: []*ResolvedPipelineRunTask{noRunStartedState[0], noRunStartedState[1]}, + }, { + name: "one-run-started-both-candidates", + state: oneRunStartedState, + candidates: sets.NewString("mytask13", "mytask14"), + expectedNext: []*ResolvedPipelineRunTask{oneRunStartedState[1]}, + }, { + name: "one-run-failed-both-candidates", + state: oneRunFailedState, + candidates: sets.NewString("mytask13", "mytask14"), + expectedNext: []*ResolvedPipelineRunTask{oneRunFailedState[1]}, }} for _, tc := range tcs { t.Run(tc.name, func(t *testing.T) { @@ -454,10 +516,22 @@ func TestPipelineRunState_SuccessfulOrSkippedDAGTasks(t *testing.T) { "not skipped since it failed", state: conditionCheckFailedWithOthersFailedState, expectedNames: []string{pts[5].Name}, + }, { + name: "one-run-started", + state: oneRunStartedState, + expectedNames: []string{}, + }, { + name: "one-run-finished", + state: oneRunFinishedState, + expectedNames: []string{pts[12].Name}, + }, { + name: "one-run-failed", + state: oneRunFailedState, + expectedNames: []string{pts[13].Name}, }} for _, tc := range tcs { t.Run(tc.name, func(t *testing.T) { - d, err := DagFromState(tc.state) + d, err := dagFromState(tc.state) if err != nil { t.Fatalf("Unexpected error while buildig DAG for state %v: %v", tc.state, err) } @@ -678,6 +752,20 @@ func TestGetPipelineConditionStatus(t *testing.T) { }, }} + var cancelledRun = PipelineRunState{{ + PipelineTask: &pts[12], + RunName: "pipelinerun-mytask13", + Run: &v1alpha1.Run{ + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{Conditions: []apis.Condition{{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionFalse, + Reason: v1alpha1.RunReasonCancelled, + }}}, + }, + }, + }} + // 6 Tasks, 4 that run in parallel in the beginning // Of the 4, 1 passed, 1 cancelled, 2 failed // 1 runAfter the passed one, currently running @@ -878,11 +966,17 @@ func TestGetPipelineConditionStatus(t *testing.T) { expectedStatus: corev1.ConditionFalse, expectedReason: v1beta1.PipelineRunReasonCancelled.String(), expectedCancelled: 1, + }, { + name: "cancelled run should result in cancelled pipeline", + state: cancelledRun, + expectedStatus: corev1.ConditionFalse, + expectedReason: v1beta1.PipelineRunReasonCancelled.String(), + expectedCancelled: 1, }} for _, tc := range tcs { t.Run(tc.name, func(t *testing.T) { pr := tb.PipelineRun("somepipelinerun") - d, err := DagFromState(tc.state) + d, err := dagFromState(tc.state) if err != nil { t.Fatalf("Unexpected error while buildig DAG for state %v: %v", tc.state, err) } @@ -1024,7 +1118,7 @@ func TestGetPipelineConditionStatus_WithFinalTasks(t *testing.T) { // pipeline should result in timeout if its runtime exceeds its spec.Timeout based on its status.Timeout func TestGetPipelineConditionStatus_PipelineTimeouts(t *testing.T) { - d, err := DagFromState(oneFinishedState) + d, err := dagFromState(oneFinishedState) if err != nil { t.Fatalf("Unexpected error while buildig DAG for state %v: %v", oneFinishedState, err) } @@ -1120,6 +1214,30 @@ func TestAdjustStartTime(t *testing.T) { }}, // We expect this to adjust to the earlier time. want: baseline.Time.Add(-2 * time.Second), + }, { + name: "run starts later", + prs: PipelineRunState{{ + Run: &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{ + Name: "blah", + CreationTimestamp: metav1.Time{Time: baseline.Time.Add(1 * time.Second)}, + }, + }, + }}, + // Stay where you are, you are before the Run. + want: baseline.Time, + }, { + name: "run starts earlier", + prs: PipelineRunState{{ + Run: &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{ + Name: "blah", + CreationTimestamp: metav1.Time{Time: baseline.Time.Add(-1 * time.Second)}, + }, + }, + }}, + // We expect this to adjust to the earlier time. + want: baseline.Time.Add(-1 * time.Second), }} for _, test := range tests { diff --git a/pkg/reconciler/pipelinerun/resources/resultrefresolution.go b/pkg/reconciler/pipelinerun/resources/resultrefresolution.go index 1a43afca6f2..3daee1e0be5 100644 --- a/pkg/reconciler/pipelinerun/resources/resultrefresolution.go +++ b/pkg/reconciler/pipelinerun/resources/resultrefresolution.go @@ -20,6 +20,7 @@ import ( "fmt" "sort" + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" "knative.dev/pkg/apis" ) @@ -34,6 +35,7 @@ type ResolvedResultRef struct { Value v1beta1.ArrayOrString ResultReference v1beta1.ResultRef FromTaskRun string + FromRun string } // ResolveResultRefs resolves any ResultReference that are found in the target ResolvedPipelineRunTask @@ -206,82 +208,126 @@ func convertPipelineResultToResultRefs(pipelineStatus v1beta1.PipelineRunStatus, } func resolveResultRef(pipelineState PipelineRunState, resultRef *v1beta1.ResultRef) (*ResolvedResultRef, error) { - referencedTaskRun, err := getReferencedTaskRun(pipelineState, resultRef) - if err != nil { - return nil, err + + referencedPipelineTask := pipelineState.ToMap()[resultRef.PipelineTask] + if referencedPipelineTask == nil { + return nil, fmt.Errorf("could not find task %q referenced by result", resultRef.PipelineTask) } - result, err := findTaskResultForParam(referencedTaskRun, resultRef) - if err != nil { - return nil, err + if !referencedPipelineTask.IsSuccessful() { + return nil, fmt.Errorf("task %q referenced by result was not successful", referencedPipelineTask.PipelineTask.Name) + } + + var runName, taskRunName, resultValue string + var err error + if referencedPipelineTask.IsCustomTask() { + runName = referencedPipelineTask.Run.Name + resultValue, err = findRunResultForParam(referencedPipelineTask.Run, resultRef) + if err != nil { + return nil, err + } + } else { + taskRunName = referencedPipelineTask.TaskRun.Name + resultValue, err = findTaskResultForParam(referencedPipelineTask.TaskRun, resultRef) + if err != nil { + return nil, err + } } + return &ResolvedResultRef{ - Value: *v1beta1.NewArrayOrString(result.Value), - FromTaskRun: referencedTaskRun.Name, + Value: *v1beta1.NewArrayOrString(resultValue), + FromTaskRun: taskRunName, + FromRun: runName, ResultReference: *resultRef, }, nil } func resolveResultRefForPipelineResult(pipelineStatus v1beta1.PipelineRunStatus, resultRef *v1beta1.ResultRef) (*ResolvedResultRef, error) { - taskRunStatus, taskRunName, err := getTaskRunStatus(pipelineStatus, resultRef.PipelineTask) - if err != nil { - return nil, err + var runName, taskRunName, resultValue string + var taskRunStatus *v1beta1.TaskRunStatus + var runStatus *v1alpha1.RunStatus + var err error + for key, taskRun := range pipelineStatus.PipelineRunStatusFields.TaskRuns { + if taskRun.PipelineTaskName == resultRef.PipelineTask { + taskRunName = key + taskRunStatus = taskRun.Status + break + } } - result, err := findTaskResultForPipelineResult(taskRunStatus, resultRef) - if err != nil { - return nil, err + if taskRunStatus != nil { + if taskRunStatus.Status.GetCondition(apis.ConditionSucceeded) == nil { + return nil, fmt.Errorf("could not find a successful task run status for task %q referenced by result", resultRef.PipelineTask) + } + resultValue, err = findTaskResultForPipelineResult(taskRunStatus, resultRef) + if err != nil { + return nil, err + } + } else { + for key, run := range pipelineStatus.PipelineRunStatusFields.Runs { + if run.PipelineTaskName == resultRef.PipelineTask { + runName = key + runStatus = run.Status + break + } + } + if runStatus != nil { + if runStatus.Status.GetCondition(apis.ConditionSucceeded) == nil { + return nil, fmt.Errorf("could not find a successful run status for task %q referenced by result", resultRef.PipelineTask) + } + resultValue, err = findRunResultForPipelineResult(runStatus, resultRef) + if err != nil { + return nil, err + } + } else { + return nil, fmt.Errorf("could not find task run or run status for task %q referenced by result", resultRef.PipelineTask) + } } + return &ResolvedResultRef{ - Value: *v1beta1.NewArrayOrString(result.Value), + Value: *v1beta1.NewArrayOrString(resultValue), FromTaskRun: taskRunName, + FromRun: runName, ResultReference: *resultRef, }, nil } -func getReferencedTaskRun(pipelineState PipelineRunState, reference *v1beta1.ResultRef) (*v1beta1.TaskRun, error) { - referencedPipelineTask := pipelineState.ToMap()[reference.PipelineTask] - - if referencedPipelineTask == nil { - return nil, fmt.Errorf("could not find task %q referenced by result", reference.PipelineTask) - } - if referencedPipelineTask.TaskRun == nil || referencedPipelineTask.IsFailure() { - return nil, fmt.Errorf("could not find successful taskrun for task %q", referencedPipelineTask.PipelineTask.Name) +func findRunResultForPipelineResult(runStatus *v1alpha1.RunStatus, reference *v1beta1.ResultRef) (string, error) { + for _, result := range runStatus.Results { + if result.Name == reference.Result { + return result.Value, nil + } } - return referencedPipelineTask.TaskRun, nil + return "", fmt.Errorf("Could not find result with name %s for task run %s", reference.Result, reference.PipelineTask) } -func getTaskRunStatus(pipelineStatus v1beta1.PipelineRunStatus, pipelineTaskName string) (*v1beta1.TaskRunStatus, string, error) { - for key, taskRun := range pipelineStatus.PipelineRunStatusFields.TaskRuns { - // check if the task run was successful - if taskRun.PipelineTaskName == pipelineTaskName { - c := taskRun.Status.GetCondition(apis.ConditionSucceeded) - if c == nil { - return nil, "", fmt.Errorf("could not find a successful task run status for task %q referenced by result", pipelineTaskName) - } - return taskRun.Status, key, nil +func findTaskResultForPipelineResult(taskStatus *v1beta1.TaskRunStatus, reference *v1beta1.ResultRef) (string, error) { + results := taskStatus.TaskRunStatusFields.TaskRunResults + for _, result := range results { + if result.Name == reference.Result { + return result.Value, nil } } - return nil, "", fmt.Errorf("could not find task run status for task %q referenced by result", pipelineTaskName) + return "", fmt.Errorf("Could not find result with name %s for task run %s", reference.Result, reference.PipelineTask) } -func findTaskResultForPipelineResult(taskStatus *v1beta1.TaskRunStatus, reference *v1beta1.ResultRef) (*v1beta1.TaskRunResult, error) { - results := taskStatus.TaskRunStatusFields.TaskRunResults +func findRunResultForParam(run *v1alpha1.Run, reference *v1beta1.ResultRef) (string, error) { + results := run.Status.Results for _, result := range results { if result.Name == reference.Result { - return &result, nil + return result.Value, nil } } - return nil, fmt.Errorf("Could not find result with name %s for task run %s", reference.Result, reference.PipelineTask) + return "", fmt.Errorf("Could not find result with name %s for task %s", reference.Result, reference.PipelineTask) } -func findTaskResultForParam(taskRun *v1beta1.TaskRun, reference *v1beta1.ResultRef) (*v1beta1.TaskRunResult, error) { +func findTaskResultForParam(taskRun *v1beta1.TaskRun, reference *v1beta1.ResultRef) (string, error) { results := taskRun.Status.TaskRunStatusFields.TaskRunResults for _, result := range results { if result.Name == reference.Result { - return &result, nil + return result.Value, nil } } - return nil, fmt.Errorf("Could not find result with name %s for task run %s", reference.Result, reference.PipelineTask) + return "", fmt.Errorf("Could not find result with name %s for task %s", reference.Result, reference.PipelineTask) } func (rs ResolvedResultRefs) getStringReplacements() map[string]string { diff --git a/pkg/reconciler/pipelinerun/resources/resultrefresolution_test.go b/pkg/reconciler/pipelinerun/resources/resultrefresolution_test.go index bf191744bf0..5d8df95ad39 100644 --- a/pkg/reconciler/pipelinerun/resources/resultrefresolution_test.go +++ b/pkg/reconciler/pipelinerun/resources/resultrefresolution_test.go @@ -8,15 +8,30 @@ import ( "github.com/google/go-cmp/cmp" tb "github.com/tektoncd/pipeline/internal/builder/v1beta1" + "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" "github.com/tektoncd/pipeline/test/diff" corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/selection" "knative.dev/pkg/apis" + duckv1 "knative.dev/pkg/apis/duck/v1" duckv1beta1 "knative.dev/pkg/apis/duck/v1beta1" ) +var ( + successCondition = apis.Condition{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionTrue, + } + failedCondition = apis.Condition{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionFalse, + } +) + func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { + for _, tt := range []struct { name string pipelineRunState PipelineRunState @@ -44,6 +59,7 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { pipelineRunState: PipelineRunState{{ TaskRunName: "aTaskRun", TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), tb.TaskRunResult("aResult", "aResultValue"), )), PipelineTask: &v1beta1.PipelineTask{ @@ -69,6 +85,7 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { pipelineRunState: PipelineRunState{{ TaskRunName: "aTaskRun", TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), tb.TaskRunResult("aResult", "aResultValue"), )), PipelineTask: &v1beta1.PipelineTask{ @@ -78,6 +95,7 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { }, { TaskRunName: "bTaskRun", TaskRun: tb.TaskRun("bTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), tb.TaskRunResult("bResult", "bResultValue"), )), PipelineTask: &v1beta1.PipelineTask{ @@ -110,6 +128,7 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { pipelineRunState: PipelineRunState{{ TaskRunName: "aTaskRun", TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), tb.TaskRunResult("aResult", "aResultValue"), )), PipelineTask: &v1beta1.PipelineTask{ @@ -134,7 +153,9 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { name: "unsuccessful resolution: referenced result doesn't exist in referenced task", pipelineRunState: PipelineRunState{{ TaskRunName: "aTaskRun", - TaskRun: tb.TaskRun("aTaskRun"), + TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), + )), PipelineTask: &v1beta1.PipelineTask{ Name: "aTask", TaskRef: &v1beta1.TaskRef{Name: "aTask"}, @@ -169,13 +190,98 @@ func TestTaskParamResolver_ResolveResultRefs(t *testing.T) { }, want: nil, wantErr: true, + }, { + name: "failed resolution: using result reference to a failed task", + pipelineRunState: PipelineRunState{{ + TaskRunName: "aTaskRun", + TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(failedCondition), + )), + PipelineTask: &v1beta1.PipelineTask{ + Name: "aTask", + TaskRef: &v1beta1.TaskRef{Name: "aTask"}, + }, + }}, + param: v1beta1.Param{ + Name: "targetParam", + Value: *v1beta1.NewArrayOrString("$(tasks.aTask.results.aResult)"), + }, + want: nil, + wantErr: true, + }, { + name: "successful resolution: using result reference to a Run", + pipelineRunState: PipelineRunState{{ + RunName: "aRun", + Run: &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{Name: "aRun"}, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{successCondition}, + }, + RunStatusFields: v1alpha1.RunStatusFields{ + Results: []v1alpha1.RunResult{{ + Name: "aResult", + Value: "aResultValue", + }}, + }, + }, + }, + PipelineTask: &v1beta1.PipelineTask{ + Name: "aCustomPipelineTask", + TaskRef: &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example", Name: "aTask"}, + }, + }}, + param: v1beta1.Param{ + Name: "targetParam", + Value: *v1beta1.NewArrayOrString("$(tasks.aCustomPipelineTask.results.aResult)"), + }, + want: ResolvedResultRefs{{ + Value: *v1beta1.NewArrayOrString("aResultValue"), + ResultReference: v1beta1.ResultRef{ + PipelineTask: "aCustomPipelineTask", + Result: "aResult", + }, + FromRun: "aRun", + }}, + wantErr: false, + }, { + name: "failed resolution: using result reference to a failed Run", + pipelineRunState: PipelineRunState{{ + RunName: "aRun", + Run: &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{Name: "aRun"}, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{failedCondition}, + }, + }, + }, + PipelineTask: &v1beta1.PipelineTask{ + Name: "aCustomPipelineTask", + TaskRef: &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example", Name: "aTask"}, + }, + }}, + param: v1beta1.Param{ + Name: "targetParam", + Value: *v1beta1.NewArrayOrString("$(tasks.aCustomPipelineTask.results.aResult)"), + }, + want: nil, + wantErr: true, }} { t.Run(tt.name, func(t *testing.T) { t.Logf("test name: %s\n", tt.name) got, err := extractResultRefsForParam(tt.pipelineRunState, tt.param) // sort result ref based on task name to guarantee an certain order sort.SliceStable(got, func(i, j int) bool { - return strings.Compare(got[i].FromTaskRun, got[j].FromTaskRun) < 0 + fromI := got[i].FromTaskRun + if fromI == "" { + fromI = got[i].FromRun + } + fromJ := got[j].FromTaskRun + if fromJ == "" { + fromJ = got[j].FromRun + } + return strings.Compare(fromI, fromJ) < 0 }) if (err != nil) != tt.wantErr { t.Fatalf("ResolveResultRef() error = %v, wantErr %v", err, tt.wantErr) @@ -210,6 +316,7 @@ func TestResolveResultRefs(t *testing.T) { pipelineRunState := PipelineRunState{{ TaskRunName: "aTaskRun", TaskRun: tb.TaskRun("aTaskRun", tb.TaskRunStatus( + tb.StatusCondition(successCondition), tb.TaskRunResult("aResult", "aResultValue"), )), PipelineTask: &v1beta1.PipelineTask{ @@ -254,6 +361,35 @@ func TestResolveResultRefs(t *testing.T) { Value: *v1beta1.NewArrayOrString("$(tasks.aTask.results.missingResult)"), }}, }, + }, { + RunName: "aRun", + Run: &v1alpha1.Run{ + ObjectMeta: metav1.ObjectMeta{Name: "aRun"}, + Status: v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{successCondition}, + }, + RunStatusFields: v1alpha1.RunStatusFields{ + Results: []v1alpha1.RunResult{{ + Name: "aResult", + Value: "aResultValue", + }}, + }, + }, + }, + PipelineTask: &v1beta1.PipelineTask{ + Name: "aCustomPipelineTask", + TaskRef: &v1beta1.TaskRef{APIVersion: "example.dev/v0", Kind: "Example", Name: "aTask"}, + }, + }, { + PipelineTask: &v1beta1.PipelineTask{ + Name: "bTask", + TaskRef: &v1beta1.TaskRef{Name: "bTask"}, + Params: []v1beta1.Param{{ + Name: "bParam", + Value: *v1beta1.NewArrayOrString("$(tasks.aCustomPipelineTask.results.aResult)"), + }}, + }, }} for _, tt := range []struct { @@ -316,11 +452,34 @@ func TestResolveResultRefs(t *testing.T) { }, want: nil, wantErr: true, + }, { + name: "Test successful result references resolution - params - Run", + pipelineRunState: pipelineRunState, + targets: PipelineRunState{ + pipelineRunState[6], + }, + want: ResolvedResultRefs{{ + Value: *v1beta1.NewArrayOrString("aResultValue"), + ResultReference: v1beta1.ResultRef{ + PipelineTask: "aCustomPipelineTask", + Result: "aResult", + }, + FromRun: "aRun", + }}, + wantErr: false, }} { t.Run(tt.name, func(t *testing.T) { got, err := ResolveResultRefs(tt.pipelineRunState, tt.targets) sort.SliceStable(got, func(i, j int) bool { - return strings.Compare(got[i].FromTaskRun, got[j].FromTaskRun) < 0 + fromI := got[i].FromTaskRun + if fromI == "" { + fromI = got[i].FromRun + } + fromJ := got[j].FromTaskRun + if fromJ == "" { + fromJ = got[j].FromRun + } + return strings.Compare(fromI, fromJ) < 0 }) if (err != nil) != tt.wantErr { t.Errorf("ResolveResultRefs() error = %v, wantErr %v", err, tt.wantErr) @@ -335,6 +494,7 @@ func TestResolveResultRefs(t *testing.T) { func TestResolvePipelineResultRefs(t *testing.T) { taskrunStatus := map[string]*v1beta1.PipelineRunTaskRunStatus{} + taskrunStatus["aTaskRun"] = &v1beta1.PipelineRunTaskRunStatus{ PipelineTaskName: "aTask", Status: &v1beta1.TaskRunStatus{ @@ -376,9 +536,35 @@ func TestResolvePipelineResultRefs(t *testing.T) { }, }, } + + runStatus := map[string]*v1beta1.PipelineRunRunStatus{} + runStatus["aRun"] = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: "CustomPipelineTaskA", + Status: &v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{successCondition}, + }, + RunStatusFields: v1alpha1.RunStatusFields{ + Results: []v1alpha1.RunResult{{ + Name: "aResult", + Value: "aResultValue", + }}, + }, + }, + } + runStatus["bRun"] = &v1beta1.PipelineRunRunStatus{ + PipelineTaskName: "CustomPipelineTaskB", + Status: &v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: []apis.Condition{failedCondition}, + }, + }, + } + status := v1beta1.PipelineRunStatus{ PipelineRunStatusFields: v1beta1.PipelineRunStatusFields{ TaskRuns: taskrunStatus, + Runs: runStatus, }, } @@ -427,13 +613,41 @@ func TestResolvePipelineResultRefs(t *testing.T) { }, FromTaskRun: "aTaskRun", }}, + }, { + name: "Test pipeline result from a successful run and failed run", + status: status, + pipelineResults: []v1beta1.PipelineResult{{ + Name: "from-CustomPipelineTaskA", + Value: "$(tasks.CustomPipelineTaskA.results.aResult)", + Description: "a result from aCustomPipelineTask", + }, { + Name: "from-CustomPipelineTaskB", + Value: "$(tasks.CustomPipelineTaskB.results.aResult)", + Description: "a result from aCustomPipelineTask", + }}, + want: ResolvedResultRefs{{ + Value: *v1beta1.NewArrayOrString("aResultValue"), + ResultReference: v1beta1.ResultRef{ + PipelineTask: "CustomPipelineTaskA", + Result: "aResult", + }, + FromRun: "aRun", + }}, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { got := ResolvePipelineResultRefs(tt.status, tt.pipelineResults) sort.SliceStable(got, func(i, j int) bool { - return strings.Compare(got[i].FromTaskRun, got[j].FromTaskRun) < 0 + fromI := got[i].FromTaskRun + if fromI == "" { + fromI = got[i].FromRun + } + fromJ := got[j].FromTaskRun + if fromJ == "" { + fromJ = got[j].FromRun + } + return strings.Compare(fromI, fromJ) < 0 }) if d := cmp.Diff(tt.want, got); d != "" { t.Fatalf("ResolveResultRef %s", diff.PrintWantGot(d)) diff --git a/test/controller.go b/test/controller.go index eff94b881b1..2ec9d2e5f16 100644 --- a/test/controller.go +++ b/test/controller.go @@ -20,8 +20,9 @@ import ( "context" "fmt" "sync/atomic" - "testing" // Link in the fakes so they get injected into injection.Fake + "testing" + // Link in the fakes so they get injected into injection.Fake "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" fakepipelineclientset "github.com/tektoncd/pipeline/pkg/client/clientset/versioned/fake" @@ -68,6 +69,7 @@ type Data struct { ClusterTasks []*v1beta1.ClusterTask PipelineResources []*v1alpha1.PipelineResource Conditions []*v1alpha1.Condition + Runs []*v1alpha1.Run Pods []*corev1.Pod Namespaces []*corev1.Namespace ConfigMaps []*corev1.ConfigMap @@ -229,6 +231,13 @@ func SeedTestData(t *testing.T, ctx context.Context, d Data) (Clients, Informers t.Fatal(err) } } + c.Pipeline.PrependReactor("*", "runs", AddToInformer(t, i.Run.Informer().GetIndexer())) + for _, run := range d.Runs { + run := run.DeepCopy() // Avoid assumptions that the informer's copy is modified. + if _, err := c.Pipeline.TektonV1alpha1().Runs(run.Namespace).Create(ctx, run, metav1.CreateOptions{}); err != nil { + t.Fatal(err) + } + } c.Kube.PrependReactor("*", "pods", AddToInformer(t, i.Pod.Informer().GetIndexer())) for _, p := range d.Pods { p := p.DeepCopy() // Avoid assumptions that the informer's copy is modified. diff --git a/test/custom_task_test.go b/test/custom_task_test.go index c1bacae8be8..a464b406c84 100644 --- a/test/custom_task_test.go +++ b/test/custom_task_test.go @@ -19,11 +19,14 @@ limitations under the License. package test import ( + "context" "testing" "time" + "github.com/google/go-cmp/cmp" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1alpha1" "github.com/tektoncd/pipeline/pkg/apis/pipeline/v1beta1" + "github.com/tektoncd/pipeline/test/diff" corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "knative.dev/pkg/apis" @@ -37,36 +40,61 @@ const ( ) func TestCustomTask(t *testing.T) { - c, namespace := setup(t) - knativetest.CleanupOnInterrupt(func() { tearDown(t, c, namespace) }, t.Logf) - defer tearDown(t, c, namespace) + ctx := context.Background() + ctx, cancel := context.WithCancel(ctx) + defer cancel() + c, namespace := setup(ctx, t) + knativetest.CleanupOnInterrupt(func() { tearDown(ctx, t, c, namespace) }, t.Logf) + defer tearDown(ctx, t, c, namespace) // Create a PipelineRun that runs a Custom Task. pipelineRunName := "custom-task-pipeline" - if _, err := c.PipelineRunClient.Create(&v1beta1.PipelineRun{ - ObjectMeta: metav1.ObjectMeta{Name: pipelineRunName}, - Spec: v1beta1.PipelineRunSpec{ - PipelineSpec: &v1beta1.PipelineSpec{ - Tasks: []v1beta1.PipelineTask{{ - Name: "example", - TaskRef: &v1beta1.TaskRef{ - APIVersion: apiVersion, - Kind: kind, - }, - }}, + if _, err := c.PipelineRunClient.Create( + ctx, + &v1beta1.PipelineRun{ + ObjectMeta: metav1.ObjectMeta{Name: pipelineRunName}, + Spec: v1beta1.PipelineRunSpec{ + PipelineSpec: &v1beta1.PipelineSpec{ + Tasks: []v1beta1.PipelineTask{{ + Name: "custom-task", + TaskRef: &v1beta1.TaskRef{ + APIVersion: apiVersion, + Kind: kind, + }, + }, { + Name: "result-consumer", + Params: []v1beta1.Param{{ + Name: "input-result-from-custom-task", Value: *v1beta1.NewArrayOrString("$(tasks.custom-task.results.runResult)"), + }}, + TaskSpec: &v1beta1.EmbeddedTask{TaskSpec: v1beta1.TaskSpec{ + Params: []v1beta1.ParamSpec{{ + Name: "input-result-from-custom-task", Type: v1beta1.ParamTypeString, + }}, + Steps: []v1beta1.Step{{Container: corev1.Container{ + Image: "ubuntu", + Command: []string{"/bin/bash"}, + Args: []string{"-c", "echo $(input-result-from-custom-task)"}, + }}}, + }}, + }}, + Results: []v1beta1.PipelineResult{{ + Name: "prResult", + Value: "$(tasks.custom-task.results.runResult)", + }}, + }, }, }, - }); err != nil { + metav1.CreateOptions{}); err != nil { t.Fatalf("Failed to create PipelineRun %q: %v", pipelineRunName, err) } // Wait for the PipelineRun to start. - if err := WaitForPipelineRunState(c, pipelineRunName, time.Minute, Running(pipelineRunName), "PipelineRunRunning"); err != nil { + if err := WaitForPipelineRunState(ctx, c, pipelineRunName, time.Minute, Running(pipelineRunName), "PipelineRunRunning"); err != nil { t.Fatalf("Waiting for PipelineRun to start running: %v", err) } // Get the status of the PipelineRun. - pr, err := c.PipelineRunClient.Get(pipelineRunName, metav1.GetOptions{}) + pr, err := c.PipelineRunClient.Get(ctx, pipelineRunName, metav1.GetOptions{}) if err != nil { t.Fatalf("Failed to get PipelineRun %q: %v", pipelineRunName, err) } @@ -82,7 +110,7 @@ func TestCustomTask(t *testing.T) { } // Get the Run. - r, err := c.RunClient.Get(runName, metav1.GetOptions{}) + r, err := c.RunClient.Get(ctx, runName, metav1.GetOptions{}) if err != nil { t.Fatalf("Failed to get Run %q: %v", runName, err) } @@ -90,20 +118,28 @@ func TestCustomTask(t *testing.T) { t.Fatalf("Run unexpectedly done: %v", r.Status.GetCondition(apis.ConditionSucceeded)) } - // Simulate a Custom Task controller updating the Run to - // done/successful. - r.Status = v1alpha1.RunStatus{Status: duckv1.Status{ - Conditions: duckv1.Conditions{{ - Type: apis.ConditionSucceeded, - Status: corev1.ConditionTrue, - }}, - }} - if _, err := c.RunClient.UpdateStatus(r); err != nil { + // Simulate a Custom Task controller updating the Run to done/successful. + r.Status = v1alpha1.RunStatus{ + Status: duckv1.Status{ + Conditions: duckv1.Conditions{{ + Type: apis.ConditionSucceeded, + Status: corev1.ConditionTrue, + }}, + }, + RunStatusFields: v1alpha1.RunStatusFields{ + Results: []v1alpha1.RunResult{{ + Name: "runResult", + Value: "aResultValue", + }}, + }, + } + + if _, err := c.RunClient.UpdateStatus(ctx, r, metav1.UpdateOptions{}); err != nil { t.Fatalf("Failed to update Run to successful: %v", err) } // Get the Run. - r, err = c.RunClient.Get(runName, metav1.GetOptions{}) + r, err = c.RunClient.Get(ctx, runName, metav1.GetOptions{}) if err != nil { t.Fatalf("Failed to get Run %q: %v", runName, err) } @@ -112,7 +148,57 @@ func TestCustomTask(t *testing.T) { } // Wait for the PipelineRun to become done/successful. - if err := WaitForPipelineRunState(c, pipelineRunName, time.Minute, PipelineRunSucceed(pipelineRunName), "PipelineRunCompleted"); err != nil { + if err := WaitForPipelineRunState(ctx, c, pipelineRunName, time.Minute, PipelineRunSucceed(pipelineRunName), "PipelineRunCompleted"); err != nil { t.Fatalf("Waiting for PipelineRun to complete successfully: %v", err) } + + // Get the updated status of the PipelineRun. + pr, err = c.PipelineRunClient.Get(ctx, pipelineRunName, metav1.GetOptions{}) + if err != nil { + t.Fatalf("Failed to get PipelineRun %q after it completed: %v", pipelineRunName, err) + } + + // Get the TaskRun name. + if len(pr.Status.TaskRuns) != 1 { + t.Fatalf("PipelineRun had unexpected .status.taskRuns; got %d, want 1", len(pr.Status.TaskRuns)) + } + var taskRunName string + for k := range pr.Status.TaskRuns { + taskRunName = k + break + } + + // Get the TaskRun. + taskRun, err := c.TaskRunClient.Get(ctx, taskRunName, metav1.GetOptions{}) + if err != nil { + t.Fatalf("Failed to get TaskRun %q: %v", taskRunName, err) + } + + // Validate the task's result reference to the custom task's result was resolved. + expectedTaskRunParams := []v1beta1.Param{{ + Name: "input-result-from-custom-task", Value: *v1beta1.NewArrayOrString("aResultValue"), + }} + + if d := cmp.Diff(expectedTaskRunParams, taskRun.Spec.Params); d != "" { + t.Fatalf("Unexpected TaskRun Params: %s", diff.PrintWantGot(d)) + } + + // Validate that the pipeline's result reference to the custom task's result was resolved. + // This validation has been commented out because of a race condition. + // The pipelinerun reconciler updates the pipelinerun results AFTER the pipelinerun is marked DONE. + // So when the test case sees the pipelinerun is done, the results may or may not have been + // updated yet. The FUBAR logic in the pipelinerun reconciler needs to be fixed to not mark + // the PR done before all the status updates are made. + + // expectedPipelineResults := []v1beta1.PipelineRunResult{{ + // Name: "prResult", + // Value: "aResultValue", + // }} + + // if len(pr.Status.PipelineResults) == 0 { + // t.Fatalf("Expected PipelineResults but there are none") + // } + // if d := cmp.Diff(expectedPipelineResults, pr.Status.PipelineResults); d != "" { + // t.Fatalf("Unexpected PipelineResults: %s", diff.PrintWantGot(d)) + // } } diff --git a/third_party/github.com/docker/cli/cli/config/NOTICE b/third_party/github.com/docker/cli/cli/config/NOTICE index 58b19b6d15b..0c74e15b057 100644 --- a/third_party/github.com/docker/cli/cli/config/NOTICE +++ b/third_party/github.com/docker/cli/cli/config/NOTICE @@ -3,7 +3,7 @@ Copyright 2012-2017 Docker, Inc. This product includes software developed at Docker, Inc. (https://www.docker.com). -This product contains software (https://github.com/creack/pty) developed +This product contains software (https://github.com/kr/pty) developed by Keith Rarick, licensed under the MIT License. The following is courtesy of our legal counsel: diff --git a/vendor/github.com/docker/cli/AUTHORS b/vendor/github.com/docker/cli/AUTHORS index ecb6251ba0d..04edcf794e2 100644 --- a/vendor/github.com/docker/cli/AUTHORS +++ b/vendor/github.com/docker/cli/AUTHORS @@ -11,7 +11,6 @@ Abin Shahab Ace Tang Addam Hardy Adolfo Ochagavía -Adrian Plata Adrien Duermael Adrien Folie Ahmet Alp Balkan @@ -137,7 +136,6 @@ Dafydd Crosby dalanlan Damien Nadé Dan Cotora -Daniel Cassidy Daniel Dao Daniel Farrell Daniel Gasienica @@ -217,7 +215,6 @@ Felix Rabe Filip Jareš Flavio Crisciani Florian Klein -Forest Johnson Foysal Iqbal François Scala Fred Lifton @@ -234,7 +231,6 @@ George MacRorie George Xie Gianluca Borello Gildas Cuisinier -Goksu Toprak Gou Rao Grant Reaber Greg Pflaum @@ -355,7 +351,6 @@ Kara Alexandra Kareem Khazem Karthik Nayak Kat Samperi -Kathryn Spiers Katie McLaughlin Ke Xu Kei Ohmura @@ -377,6 +372,7 @@ Krasi Georgiev Kris-Mikael Krister Kun Zhang Kunal Kushwaha +Kyle Spiers Lachlan Cooper Lai Jiangshan Lars Kellogg-Stedman @@ -541,7 +537,6 @@ Qiang Huang Qinglan Peng qudongfang Raghavendra K T -Ravi Shekhar Jethani Ray Tsang Reficul Remy Suen @@ -558,7 +553,6 @@ Robin Naundorf Robin Speekenbrink Rodolfo Ortiz Rogelio Canedo -Rohan Verma Roland Kammerer Roman Dudin Rory Hunter @@ -707,7 +701,6 @@ Yuan Sun Yue Zhang Yunxiang Huang Zachary Romero -Zander Mackie zebrilee Zhang Kun Zhang Wei diff --git a/vendor/github.com/docker/cli/NOTICE b/vendor/github.com/docker/cli/NOTICE index 58b19b6d15b..0c74e15b057 100644 --- a/vendor/github.com/docker/cli/NOTICE +++ b/vendor/github.com/docker/cli/NOTICE @@ -3,7 +3,7 @@ Copyright 2012-2017 Docker, Inc. This product includes software developed at Docker, Inc. (https://www.docker.com). -This product contains software (https://github.com/creack/pty) developed +This product contains software (https://github.com/kr/pty) developed by Keith Rarick, licensed under the MIT License. The following is courtesy of our legal counsel: diff --git a/vendor/github.com/docker/cli/cli/config/config.go b/vendor/github.com/docker/cli/cli/config/config.go index 6e4d73dfad7..f5e33f2b3e9 100644 --- a/vendor/github.com/docker/cli/cli/config/config.go +++ b/vendor/github.com/docker/cli/cli/config/config.go @@ -106,13 +106,9 @@ func Load(configDir string) (*configfile.ConfigFile, error) { } // Can't find latest config file so check for the old one - homedir, err := os.UserHomeDir() - if err != nil { - return configFile, errors.Wrap(err, oldConfigfile) - } - confFile := filepath.Join(homedir, oldConfigfile) + confFile := filepath.Join(homedir.Get(), oldConfigfile) if _, err := os.Stat(confFile); err != nil { - return configFile, nil // missing file is not an error + return configFile, nil //missing file is not an error } file, err := os.Open(confFile) if err != nil { diff --git a/vendor/github.com/docker/cli/cli/config/configfile/file.go b/vendor/github.com/docker/cli/cli/config/configfile/file.go index a4e97a5caa6..388a5d54d69 100644 --- a/vendor/github.com/docker/cli/cli/config/configfile/file.go +++ b/vendor/github.com/docker/cli/cli/config/configfile/file.go @@ -196,9 +196,6 @@ func (configFile *ConfigFile) Save() error { os.Remove(temp.Name()) return err } - // Try copying the current config file (if any) ownership and permissions - copyFilePermissions(configFile.Filename, temp.Name()) - return os.Rename(temp.Name(), configFile.Filename) } diff --git a/vendor/github.com/docker/cli/cli/config/configfile/file_unix.go b/vendor/github.com/docker/cli/cli/config/configfile/file_unix.go deleted file mode 100644 index 3ca65c6140d..00000000000 --- a/vendor/github.com/docker/cli/cli/config/configfile/file_unix.go +++ /dev/null @@ -1,35 +0,0 @@ -// +build !windows - -package configfile - -import ( - "os" - "syscall" -) - -// copyFilePermissions copies file ownership and permissions from "src" to "dst", -// ignoring any error during the process. -func copyFilePermissions(src, dst string) { - var ( - mode os.FileMode = 0600 - uid, gid int - ) - - fi, err := os.Stat(src) - if err != nil { - return - } - if fi.Mode().IsRegular() { - mode = fi.Mode() - } - if err := os.Chmod(dst, mode); err != nil { - return - } - - uid = int(fi.Sys().(*syscall.Stat_t).Uid) - gid = int(fi.Sys().(*syscall.Stat_t).Gid) - - if uid > 0 && gid > 0 { - _ = os.Chown(dst, uid, gid) - } -} diff --git a/vendor/github.com/docker/cli/cli/config/configfile/file_windows.go b/vendor/github.com/docker/cli/cli/config/configfile/file_windows.go deleted file mode 100644 index 42fffc39ad2..00000000000 --- a/vendor/github.com/docker/cli/cli/config/configfile/file_windows.go +++ /dev/null @@ -1,5 +0,0 @@ -package configfile - -func copyFilePermissions(src, dst string) { - // TODO implement for Windows -} diff --git a/vendor/modules.txt b/vendor/modules.txt index c7f0426a672..bf085a93261 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -123,7 +123,7 @@ github.com/cloudevents/sdk-go/v2/types github.com/davecgh/go-spew/spew # github.com/dgrijalva/jwt-go v3.2.0+incompatible github.com/dgrijalva/jwt-go -# github.com/docker/cli v0.0.0-20200303162255-7d407207c304 +# github.com/docker/cli v0.0.0-20200210162036-a4bedce16568 github.com/docker/cli/cli/config github.com/docker/cli/cli/config/configfile github.com/docker/cli/cli/config/credentials