Skip to content

Commit 697ee8e

Browse files
committed
don't create output imagestrem if already exists with newapp; better circular tag detection
1 parent 577d6ce commit 697ee8e

File tree

7 files changed

+555
-56
lines changed

7 files changed

+555
-56
lines changed

pkg/generate/app/cmd/newapp.go

+171-11
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ import (
2020
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
2121
"k8s.io/apimachinery/pkg/runtime"
2222
kutilerrors "k8s.io/apimachinery/pkg/util/errors"
23+
"k8s.io/apimachinery/pkg/util/sets"
2324
kapi "k8s.io/kubernetes/pkg/api"
2425
"k8s.io/kubernetes/pkg/api/validation"
2526
kclientset "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
@@ -816,7 +817,8 @@ func (c *AppConfig) Run() (*AppResult, error) {
816817
return nil, err
817818
}
818819

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)}
820822
objects := app.Objects{}
821823
accept := app.NewAcceptFirst()
822824
for _, p := range pipelines {
@@ -914,6 +916,113 @@ func (c *AppConfig) Run() (*AppResult, error) {
914916
}, nil
915917
}
916918

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+
9171026
// followRefToDockerImage follows a buildconfig...To/From reference until it
9181027
// terminates in docker image information. This can include dereferencing chains
9191028
// of ImageStreamTag references that already exist or which are being created.
@@ -979,27 +1088,29 @@ func (c *AppConfig) followRefToDockerImage(ref *kapi.ObjectReference, isContext
9791088

9801089
// The imagestream is usually being created alongside the buildconfig
9811090
// 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 {
9921092
var err error
9931093
isContext, err = c.ImageClient.ImageStreams(isNS).Get(isName, metav1.GetOptions{})
9941094
if err != nil {
9951095
return nil, fmt.Errorf("Unable to check for circular build input/outputs: %v", err)
9961096
}
1097+
if isContext == nil {
1098+
return nil, nil
1099+
}
9971100
}
9981101
}
9991102

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+
10001110
// Dereference ImageStreamTag to see what it is pointing to
10011111
target := isContext.Spec.Tags[isTag].From
10021112

1113+
// From can still be nil even with the call to FollowTagReference returning OK
10031114
if target == nil {
10041115
if isContext.Spec.DockerImageRepository == "" {
10051116
// 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 {
10381149

10391150
dockerInput, err := c.followRefToDockerImage(input, nil, objects)
10401151
if err != nil {
1152+
if _, ok := err.(app.CircularReferenceError); ok {
1153+
return err
1154+
}
10411155
glog.Warningf("Unable to check for circular build input: %v", err)
10421156
return nil
10431157
}
10441158
glog.V(5).Infof("Post follow input:\n%#v\n", dockerInput)
10451159

10461160
dockerOutput, err := c.followRefToDockerImage(output, nil, objects)
10471161
if err != nil {
1162+
if _, ok := err.(app.CircularReferenceError); ok {
1163+
return err
1164+
}
10481165
glog.Warningf("Unable to check for circular build output: %v", err)
10491166
return nil
10501167
}
10511168
glog.V(5).Infof("Post follow:\n%#v\n", dockerOutput)
10521169

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+
10531213
if dockerInput != nil && dockerOutput != nil {
10541214
if reflect.DeepEqual(dockerInput, dockerOutput) {
10551215
return app.CircularOutputReferenceError{Reference: fmt.Sprintf("%s", dockerInput.Name)}

0 commit comments

Comments
 (0)