@@ -20,6 +20,7 @@ import (
20
20
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
21
21
"k8s.io/apimachinery/pkg/runtime"
22
22
kutilerrors "k8s.io/apimachinery/pkg/util/errors"
23
+ "k8s.io/apimachinery/pkg/util/sets"
23
24
kapi "k8s.io/kubernetes/pkg/api"
24
25
"k8s.io/kubernetes/pkg/api/validation"
25
26
kclientset "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
@@ -816,7 +817,8 @@ func (c *AppConfig) Run() (*AppResult, error) {
816
817
return nil , err
817
818
}
818
819
819
- acceptors := app.Acceptors {app .NewAcceptUnique (c .Typer ), app .AcceptNew }
820
+ acceptors := app.Acceptors {app .NewAcceptUnique (c .Typer ), app .AcceptNew ,
821
+ app .NewAcceptNonExistentImageStream (c .Typer , c .ImageClient , c .OriginNamespace ), app .NewAcceptNonExistentImageStreamTag (c .Typer , c .ImageClient , c .OriginNamespace )}
820
822
objects := app.Objects {}
821
823
accept := app .NewAcceptFirst ()
822
824
for _ , p := range pipelines {
@@ -914,6 +916,113 @@ func (c *AppConfig) Run() (*AppResult, error) {
914
916
}, nil
915
917
}
916
918
919
+ func (c * AppConfig ) findImageStreamInObjectList (objects app.Objects , name , namespace string ) * imageapi.ImageStream {
920
+ for _ , check := range objects {
921
+ if is , ok := check .(* imageapi.ImageStream ); ok {
922
+ nsToCompare := is .Namespace
923
+ if len (nsToCompare ) == 0 {
924
+ nsToCompare = c .OriginNamespace
925
+ }
926
+ if is .Name == name && nsToCompare == namespace {
927
+ return is
928
+ }
929
+ }
930
+ }
931
+ return nil
932
+ }
933
+
934
+ // crossStreamCircularTagReference inherits some logic from imageapi.FollowTagReference, but differs in that a) it is only concerned
935
+ // with whether we can definitively say the IST chain is circular, and b) can cross image stream boundaries;
936
+ // not in imageapi pkg (see imports above) like other helpers cause of import cycle with the image client
937
+ func (c * AppConfig ) crossStreamCircularTagReference (stream * imageapi.ImageStream , tag string , objects app.Objects ) bool {
938
+ if stream == nil {
939
+ return false
940
+ }
941
+ seen := sets .NewString ()
942
+ for {
943
+ if seen .Has (stream .ObjectMeta .Namespace + ":" + stream .ObjectMeta .Name + ":" + tag ) {
944
+ // circular reference
945
+ return true
946
+ }
947
+ seen .Insert (stream .ObjectMeta .Namespace + ":" + stream .ObjectMeta .Name + ":" + tag )
948
+ tagRef , ok := stream .Spec .Tags [tag ]
949
+ if ! ok {
950
+ // no tag at the end of the rainbow
951
+ return false
952
+ }
953
+ if tagRef .From == nil || tagRef .From .Kind != "ImageStreamTag" {
954
+ // terminating tag
955
+ return false
956
+ }
957
+ if strings .Contains (tagRef .From .Name , ":" ) {
958
+ // another stream
959
+ fromstream , fromtag , ok := imageapi .SplitImageStreamTag (tagRef .From .Name )
960
+ if ! ok {
961
+ return false
962
+ }
963
+ stream , err := c .ImageClient .ImageStreams (tagRef .From .Namespace ).Get (fromstream , metav1.GetOptions {})
964
+ if err != nil && ! kerrors .IsNotFound (err ) {
965
+ return false
966
+ }
967
+ if stream == nil || kerrors .IsNotFound (err ) {
968
+ // in case we are creating with this generate run, check the list of objects provided before
969
+ // giving up
970
+ if stream = c .findImageStreamInObjectList (objects , fromstream , tagRef .From .Namespace ); stream == nil {
971
+ return false
972
+ }
973
+ }
974
+ tag = fromtag
975
+ } else {
976
+ tag = tagRef .From .Name
977
+ }
978
+ }
979
+ }
980
+
981
+ // crossStreamInputToOutputTagReference inherits some logic from crossStreamCircularTagReference, but differs in that a) it is only concerned
982
+ // with whether we can definitively say the input IST from/follow chain lands at the output IST;
983
+ // this method currently assumes that crossStreamCircularTagReference was called on both in/out stream:tag combinations prior to entering this method
984
+ // not in imageapi pkg (see imports above) like other helpers cause of import cycle with the image client
985
+ func (c * AppConfig ) crossStreamInputToOutputTagReference (instream , outstream * imageapi.ImageStream , intag , outtag string , objects app.Objects ) bool {
986
+ if instream == nil || outstream == nil {
987
+ return false
988
+ }
989
+ for {
990
+ if instream .ObjectMeta .Namespace == outstream .ObjectMeta .Namespace &&
991
+ instream .ObjectMeta .Name == outstream .ObjectMeta .Name &&
992
+ intag == outtag {
993
+ return true
994
+ }
995
+ tagRef , ok := instream .Spec .Tags [intag ]
996
+ if ! ok {
997
+ // no tag at the end of the rainbow
998
+ return false
999
+ }
1000
+ if tagRef .From == nil || tagRef .From .Kind != "ImageStreamTag" {
1001
+ // terminating tag
1002
+ return false
1003
+ }
1004
+ if strings .Contains (tagRef .From .Name , ":" ) {
1005
+ // another stream
1006
+ fromstream , fromtag , ok := imageapi .SplitImageStreamTag (tagRef .From .Name )
1007
+ if ! ok {
1008
+ return false
1009
+ }
1010
+ instream , err := c .ImageClient .ImageStreams (tagRef .From .Namespace ).Get (fromstream , metav1.GetOptions {})
1011
+ if err != nil && ! kerrors .IsNotFound (err ) {
1012
+ return false
1013
+ }
1014
+ if instream == nil || kerrors .IsNotFound (err ) {
1015
+ if instream = c .findImageStreamInObjectList (objects , fromstream , tagRef .From .Namespace ); instream == nil {
1016
+ return false
1017
+ }
1018
+ }
1019
+ intag = fromtag
1020
+ } else {
1021
+ intag = tagRef .From .Name
1022
+ }
1023
+ }
1024
+ }
1025
+
917
1026
// followRefToDockerImage follows a buildconfig...To/From reference until it
918
1027
// terminates in docker image information. This can include dereferencing chains
919
1028
// of ImageStreamTag references that already exist or which are being created.
@@ -979,27 +1088,29 @@ func (c *AppConfig) followRefToDockerImage(ref *kapi.ObjectReference, isContext
979
1088
980
1089
// The imagestream is usually being created alongside the buildconfig
981
1090
// when new-build is being used, so scan objects being created for it.
982
- for _ , check := range objects {
983
- if is2 , ok := check .(* imageapi.ImageStream ); ok {
984
- if is2 .Name == isName {
985
- isContext = is2
986
- break
987
- }
988
- }
989
- }
990
-
991
- if isContext == nil {
1091
+ if isContext = c .findImageStreamInObjectList (objects , isName , isNS ); isContext == nil {
992
1092
var err error
993
1093
isContext , err = c .ImageClient .ImageStreams (isNS ).Get (isName , metav1.GetOptions {})
994
1094
if err != nil {
995
1095
return nil , fmt .Errorf ("Unable to check for circular build input/outputs: %v" , err )
996
1096
}
1097
+ if isContext == nil {
1098
+ return nil , nil
1099
+ }
997
1100
}
998
1101
}
999
1102
1103
+ // protect our recursion by leveraging a non-recursive IST -> IST traversal check
1104
+ // to warn us about circular problems with the tag so we can bail
1105
+ circular := c .crossStreamCircularTagReference (isContext , isTag , objects )
1106
+ if circular {
1107
+ return nil , app.CircularReferenceError {Reference : ref .Name }
1108
+ }
1109
+
1000
1110
// Dereference ImageStreamTag to see what it is pointing to
1001
1111
target := isContext .Spec .Tags [isTag ].From
1002
1112
1113
+ // From can still be nil even with the call to FollowTagReference returning OK
1003
1114
if target == nil {
1004
1115
if isContext .Spec .DockerImageRepository == "" {
1005
1116
// Otherwise, this appears to be a new IS, created by new-app, with very little information
@@ -1038,18 +1149,67 @@ func (c *AppConfig) checkCircularReferences(objects app.Objects) error {
1038
1149
1039
1150
dockerInput , err := c .followRefToDockerImage (input , nil , objects )
1040
1151
if err != nil {
1152
+ if _ , ok := err .(app.CircularReferenceError ); ok {
1153
+ return err
1154
+ }
1041
1155
glog .Warningf ("Unable to check for circular build input: %v" , err )
1042
1156
return nil
1043
1157
}
1044
1158
glog .V (5 ).Infof ("Post follow input:\n %#v\n " , dockerInput )
1045
1159
1046
1160
dockerOutput , err := c .followRefToDockerImage (output , nil , objects )
1047
1161
if err != nil {
1162
+ if _ , ok := err .(app.CircularReferenceError ); ok {
1163
+ return err
1164
+ }
1048
1165
glog .Warningf ("Unable to check for circular build output: %v" , err )
1049
1166
return nil
1050
1167
}
1051
1168
glog .V (5 ).Infof ("Post follow:\n %#v\n " , dockerOutput )
1052
1169
1170
+ // with reuse of existing image streams, some scenarios have arisen where
1171
+ // comparisons of input/output prior to following the ref to the docker image
1172
+ // are necessary;
1173
+ // we still cite errors with circular IST chains;
1174
+ // however, if the output IST points to the same
1175
+ // image as the input IST because its referential chain takes it to the input IST (i.e. From refs),
1176
+ // where one of the ISTs has a direct docker ref, we are allowing that;
1177
+ // thus, invocations like `oc new-build --binary php`,
1178
+ // where "php" is a image stream with such a structure, do not require
1179
+ // the `--to` option
1180
+ // otherwise, we are deferring to the docker image resolution path
1181
+ if output .Kind == "ImageStreamTag" && input .Kind == "ImageStreamTag" {
1182
+ iname , itag , iok := imageapi .SplitImageStreamTag (input .Name )
1183
+ oname , otag , ook := imageapi .SplitImageStreamTag (output .Name )
1184
+ if iok && ook {
1185
+ inamespace := input .Namespace
1186
+ if len (inamespace ) == 0 {
1187
+ inamespace = c .OriginNamespace
1188
+ }
1189
+ onamespace := output .Namespace
1190
+ if len (onamespace ) == 0 {
1191
+ onamespace = c .OriginNamespace
1192
+ }
1193
+ istream , err := c .ImageClient .ImageStreams (inamespace ).Get (iname , metav1.GetOptions {})
1194
+ if istream == nil || kerrors .IsNotFound (err ) {
1195
+ istream = c .findImageStreamInObjectList (objects , iname , inamespace )
1196
+ }
1197
+ ostream , err := c .ImageClient .ImageStreams (onamespace ).Get (oname , metav1.GetOptions {})
1198
+ if ostream == nil || kerrors .IsNotFound (err ) {
1199
+ ostream = c .findImageStreamInObjectList (objects , oname , onamespace )
1200
+ }
1201
+ if istream != nil && ostream != nil {
1202
+ if circularOutput := c .crossStreamInputToOutputTagReference (istream , ostream , itag , otag , objects ); circularOutput {
1203
+ if dockerInput != nil {
1204
+ return app.CircularOutputReferenceError {Reference : fmt .Sprintf ("%s" , dockerInput .Name )}
1205
+ }
1206
+ return app.CircularOutputReferenceError {Reference : fmt .Sprintf ("%s" , input .Name )}
1207
+ }
1208
+ return nil
1209
+ }
1210
+ }
1211
+ }
1212
+
1053
1213
if dockerInput != nil && dockerOutput != nil {
1054
1214
if reflect .DeepEqual (dockerInput , dockerOutput ) {
1055
1215
return app.CircularOutputReferenceError {Reference : fmt .Sprintf ("%s" , dockerInput .Name )}
0 commit comments