github.com/DaAlbrecht/cf-cli@v0.0.0-20231128151943-1fe19bb400b9/actor/v7action/application_test.go (about)

     1  package v7action_test
     2  
     3  import (
     4  	"errors"
     5  	"fmt"
     6  	"time"
     7  
     8  	"code.cloudfoundry.org/cli/actor/actionerror"
     9  	. "code.cloudfoundry.org/cli/actor/v7action"
    10  	"code.cloudfoundry.org/cli/actor/v7action/v7actionfakes"
    11  	"code.cloudfoundry.org/cli/api/cloudcontroller/ccerror"
    12  	"code.cloudfoundry.org/cli/api/cloudcontroller/ccv3"
    13  	"code.cloudfoundry.org/cli/api/cloudcontroller/ccv3/constant"
    14  	"code.cloudfoundry.org/cli/resources"
    15  	"code.cloudfoundry.org/cli/types"
    16  	"code.cloudfoundry.org/cli/util/batcher"
    17  	"code.cloudfoundry.org/clock/fakeclock"
    18  	. "github.com/onsi/ginkgo"
    19  	. "github.com/onsi/gomega"
    20  )
    21  
    22  var _ = Describe("Application Actions", func() {
    23  	var (
    24  		actor                     *Actor
    25  		fakeCloudControllerClient *v7actionfakes.FakeCloudControllerClient
    26  		fakeConfig                *v7actionfakes.FakeConfig
    27  		fakeClock                 *fakeclock.FakeClock
    28  	)
    29  
    30  	BeforeEach(func() {
    31  		fakeCloudControllerClient = new(v7actionfakes.FakeCloudControllerClient)
    32  		fakeConfig = new(v7actionfakes.FakeConfig)
    33  		fakeClock = fakeclock.NewFakeClock(time.Now())
    34  		actor = NewActor(fakeCloudControllerClient, fakeConfig, nil, nil, nil, fakeClock)
    35  	})
    36  
    37  	Describe("DeleteApplicationByNameAndSpace", func() {
    38  		var (
    39  			warnings           Warnings
    40  			executeErr         error
    41  			deleteMappedRoutes bool
    42  			appName            string
    43  		)
    44  
    45  		JustBeforeEach(func() {
    46  			appName = "some-app"
    47  			warnings, executeErr = actor.DeleteApplicationByNameAndSpace(appName, "some-space-guid", deleteMappedRoutes)
    48  		})
    49  
    50  		When("looking up the app guid fails", func() {
    51  			BeforeEach(func() {
    52  				fakeCloudControllerClient.GetApplicationsReturns([]resources.Application{}, ccv3.Warnings{"some-get-app-warning"}, errors.New("some-get-app-error"))
    53  			})
    54  
    55  			It("returns the warnings and error", func() {
    56  				Expect(warnings).To(ConsistOf("some-get-app-warning"))
    57  				Expect(executeErr).To(MatchError("some-get-app-error"))
    58  			})
    59  		})
    60  
    61  		When("looking up the app guid succeeds without routes", func() {
    62  			BeforeEach(func() {
    63  				fakeCloudControllerClient.GetApplicationsReturns([]resources.Application{{Name: "some-app", GUID: "abc123"}}, ccv3.Warnings{"some-get-app-warning"}, nil)
    64  				deleteMappedRoutes = false
    65  			})
    66  
    67  			When("sending the delete fails", func() {
    68  				BeforeEach(func() {
    69  					fakeCloudControllerClient.DeleteApplicationReturns("", ccv3.Warnings{"some-delete-app-warning"}, errors.New("some-delete-app-error"))
    70  				})
    71  
    72  				It("returns the warnings and error", func() {
    73  					Expect(warnings).To(ConsistOf("some-get-app-warning", "some-delete-app-warning"))
    74  					Expect(executeErr).To(MatchError("some-delete-app-error"))
    75  				})
    76  			})
    77  
    78  			When("sending the delete succeeds", func() {
    79  				BeforeEach(func() {
    80  					fakeCloudControllerClient.DeleteApplicationReturns("/some-job-url", ccv3.Warnings{"some-delete-app-warning"}, nil)
    81  				})
    82  
    83  				When("polling fails", func() {
    84  					BeforeEach(func() {
    85  						fakeCloudControllerClient.PollJobReturns(ccv3.Warnings{"some-poll-warning"}, errors.New("some-poll-error"))
    86  					})
    87  
    88  					It("returns the warnings and poll error", func() {
    89  						Expect(warnings).To(ConsistOf("some-get-app-warning", "some-delete-app-warning", "some-poll-warning"))
    90  						Expect(executeErr).To(MatchError("some-poll-error"))
    91  					})
    92  				})
    93  
    94  				When("polling succeeds", func() {
    95  					BeforeEach(func() {
    96  						fakeCloudControllerClient.PollJobReturns(ccv3.Warnings{"some-poll-warning"}, nil)
    97  					})
    98  
    99  					It("returns all the warnings and no error", func() {
   100  						Expect(warnings).To(ConsistOf("some-get-app-warning", "some-delete-app-warning", "some-poll-warning"))
   101  						Expect(executeErr).ToNot(HaveOccurred())
   102  					})
   103  				})
   104  			})
   105  		})
   106  
   107  		When("looking up the app guid succeeds with routes", func() {
   108  			BeforeEach(func() {
   109  				deleteMappedRoutes = true
   110  				fakeCloudControllerClient.GetApplicationsReturns([]resources.Application{{Name: "some-app", GUID: "abc123"}}, nil, nil)
   111  			})
   112  
   113  			When("getting the routes fails", func() {
   114  				BeforeEach(func() {
   115  					fakeCloudControllerClient.GetApplicationRoutesReturns(nil, ccv3.Warnings{"get-routes-warning"}, errors.New("get-routes-error"))
   116  				})
   117  
   118  				It("returns the warnings and an error", func() {
   119  					Expect(warnings).To(ConsistOf("get-routes-warning"))
   120  					Expect(executeErr).To(MatchError("get-routes-error"))
   121  				})
   122  			})
   123  
   124  			When("getting the routes succeeds", func() {
   125  				When("there are no routes", func() {
   126  					BeforeEach(func() {
   127  						fakeCloudControllerClient.GetApplicationRoutesReturns([]resources.Route{}, nil, nil)
   128  					})
   129  
   130  					It("does not delete any routes", func() {
   131  						Expect(fakeCloudControllerClient.DeleteRouteCallCount()).To(Equal(0))
   132  					})
   133  				})
   134  
   135  				When("there are routes", func() {
   136  					BeforeEach(func() {
   137  						fakeCloudControllerClient.GetApplicationRoutesReturns([]resources.Route{{GUID: "route-1-guid"}, {GUID: "route-2-guid", URL: "route-2.example.com"}}, nil, nil)
   138  					})
   139  
   140  					It("deletes the routes", func() {
   141  						Expect(fakeCloudControllerClient.GetApplicationRoutesCallCount()).To(Equal(1))
   142  						Expect(fakeCloudControllerClient.GetApplicationRoutesArgsForCall(0)).To(Equal("abc123"))
   143  						Expect(fakeCloudControllerClient.DeleteRouteCallCount()).To(Equal(2))
   144  						guids := []string{fakeCloudControllerClient.DeleteRouteArgsForCall(0), fakeCloudControllerClient.DeleteRouteArgsForCall(1)}
   145  						Expect(guids).To(ConsistOf("route-1-guid", "route-2-guid"))
   146  					})
   147  
   148  					When("the route has already been deleted", func() {
   149  						BeforeEach(func() {
   150  							fakeCloudControllerClient.DeleteRouteReturnsOnCall(0,
   151  								"",
   152  								ccv3.Warnings{"delete-route-1-warning"},
   153  								ccerror.ResourceNotFoundError{},
   154  							)
   155  							fakeCloudControllerClient.DeleteRouteReturnsOnCall(1,
   156  								"poll-job-url",
   157  								ccv3.Warnings{"delete-route-2-warning"},
   158  								nil,
   159  							)
   160  							fakeCloudControllerClient.PollJobReturnsOnCall(1, ccv3.Warnings{"poll-job-warning"}, nil)
   161  						})
   162  
   163  						It("does **not** fail", func() {
   164  							Expect(executeErr).ToNot(HaveOccurred())
   165  							Expect(warnings).To(ConsistOf("delete-route-1-warning", "delete-route-2-warning", "poll-job-warning"))
   166  							Expect(fakeCloudControllerClient.DeleteRouteCallCount()).To(Equal(2))
   167  							Expect(fakeCloudControllerClient.PollJobCallCount()).To(Equal(2))
   168  							Expect(fakeCloudControllerClient.PollJobArgsForCall(1)).To(BeEquivalentTo("poll-job-url"))
   169  						})
   170  					})
   171  
   172  					When("app to delete has a route bound to another app", func() {
   173  						BeforeEach(func() {
   174  							fakeCloudControllerClient.GetApplicationRoutesReturns(
   175  								[]resources.Route{
   176  									{GUID: "route-1-guid"},
   177  									{GUID: "route-2-guid",
   178  										URL: "route-2.example.com",
   179  										Destinations: []resources.RouteDestination{
   180  											{App: resources.RouteDestinationApp{GUID: "abc123"}},
   181  											{App: resources.RouteDestinationApp{GUID: "different-app-guid"}},
   182  										},
   183  									},
   184  								},
   185  								nil,
   186  								nil,
   187  							)
   188  						})
   189  
   190  						It("refuses the entire operation", func() {
   191  							Expect(executeErr).To(MatchError(actionerror.RouteBoundToMultipleAppsError{AppName: "some-app", RouteURL: "route-2.example.com"}))
   192  							Expect(warnings).To(BeEmpty())
   193  							Expect(fakeCloudControllerClient.DeleteApplicationCallCount()).To(Equal(0))
   194  							Expect(fakeCloudControllerClient.DeleteRouteCallCount()).To(Equal(0))
   195  						})
   196  					})
   197  
   198  					When("deleting the route fails", func() {
   199  						BeforeEach(func() {
   200  							fakeCloudControllerClient.DeleteRouteReturnsOnCall(0,
   201  								"poll-job-url",
   202  								ccv3.Warnings{"delete-route-1-warning"},
   203  								nil,
   204  							)
   205  							fakeCloudControllerClient.DeleteRouteReturnsOnCall(1,
   206  								"",
   207  								ccv3.Warnings{"delete-route-2-warning"},
   208  								errors.New("delete-route-2-error"),
   209  							)
   210  						})
   211  
   212  						It("returns the error", func() {
   213  							Expect(executeErr).To(MatchError("delete-route-2-error"))
   214  							Expect(warnings).To(ConsistOf("delete-route-1-warning", "delete-route-2-warning"))
   215  						})
   216  					})
   217  
   218  					When("the polling job fails", func() {
   219  						BeforeEach(func() {
   220  							fakeCloudControllerClient.PollJobReturns(ccv3.Warnings{"poll-job-warning"}, errors.New("poll-job-error"))
   221  						})
   222  
   223  						It("returns the error", func() {
   224  							Expect(executeErr).To(MatchError("poll-job-error"))
   225  						})
   226  					})
   227  
   228  				})
   229  			})
   230  		})
   231  	})
   232  
   233  	Describe("GetApplicationsByGUIDs", func() {
   234  		When("all of the requested apps exist", func() {
   235  			BeforeEach(func() {
   236  				fakeCloudControllerClient.GetApplicationsReturns(
   237  					[]resources.Application{
   238  						{
   239  							Name: "some-app-name",
   240  							GUID: "some-app-guid",
   241  						},
   242  						{
   243  							Name: "other-app-name",
   244  							GUID: "other-app-guid",
   245  						},
   246  					},
   247  					ccv3.Warnings{"some-warning"},
   248  					nil,
   249  				)
   250  			})
   251  
   252  			It("returns the applications and warnings", func() {
   253  				apps, warnings, err := actor.GetApplicationsByGUIDs([]string{"some-app-guid", "other-app-guid"})
   254  				Expect(err).ToNot(HaveOccurred())
   255  				Expect(apps).To(ConsistOf(
   256  					resources.Application{
   257  						Name: "some-app-name",
   258  						GUID: "some-app-guid",
   259  					},
   260  					resources.Application{
   261  						Name: "other-app-name",
   262  						GUID: "other-app-guid",
   263  					},
   264  				))
   265  				Expect(warnings).To(ConsistOf("some-warning"))
   266  
   267  				Expect(fakeCloudControllerClient.GetApplicationsCallCount()).To(Equal(1))
   268  				Expect(fakeCloudControllerClient.GetApplicationsArgsForCall(0)).To(ConsistOf(
   269  					ccv3.Query{Key: ccv3.GUIDFilter, Values: []string{"some-app-guid", "other-app-guid"}},
   270  				))
   271  			})
   272  		})
   273  
   274  		When("at least one of the requested apps does not exist", func() {
   275  			BeforeEach(func() {
   276  				fakeCloudControllerClient.GetApplicationsReturns(
   277  					[]resources.Application{
   278  						{
   279  							Name: "some-app-name",
   280  							GUID: "some-app-guid",
   281  						},
   282  					},
   283  					ccv3.Warnings{"some-warning"},
   284  					nil,
   285  				)
   286  			})
   287  
   288  			It("returns an ApplicationNotFoundError and the warnings", func() {
   289  				_, warnings, err := actor.GetApplicationsByGUIDs([]string{"some-app-guid", "nonexistent-app-guid"})
   290  				Expect(warnings).To(ConsistOf("some-warning"))
   291  				Expect(err).To(MatchError(actionerror.ApplicationsNotFoundError{}))
   292  			})
   293  		})
   294  
   295  		When("a single app has two routes", func() {
   296  			BeforeEach(func() {
   297  				fakeCloudControllerClient.GetApplicationsReturns(
   298  					[]resources.Application{
   299  						{
   300  							Name: "some-app-name",
   301  							GUID: "some-app-guid",
   302  						},
   303  					},
   304  					ccv3.Warnings{"some-warning"},
   305  					nil,
   306  				)
   307  			})
   308  
   309  			It("returns an ApplicationNotFoundError and the warnings", func() {
   310  				_, warnings, err := actor.GetApplicationsByGUIDs([]string{"some-app-guid", "some-app-guid"})
   311  				Expect(err).ToNot(HaveOccurred())
   312  				Expect(warnings).To(ConsistOf("some-warning"))
   313  			})
   314  		})
   315  
   316  		When("the cloud controller client returns an error", func() {
   317  			var expectedError error
   318  
   319  			BeforeEach(func() {
   320  				expectedError = errors.New("I am a CloudControllerClient Error")
   321  				fakeCloudControllerClient.GetApplicationsReturns(
   322  					[]resources.Application{},
   323  					ccv3.Warnings{"some-warning"},
   324  					expectedError)
   325  			})
   326  
   327  			It("returns the warnings and the error", func() {
   328  				_, warnings, err := actor.GetApplicationsByGUIDs([]string{"some-app-guid"})
   329  				Expect(warnings).To(ConsistOf("some-warning"))
   330  				Expect(err).To(MatchError(expectedError))
   331  			})
   332  		})
   333  
   334  		When("there are many guids", func() {
   335  			const batches = 10
   336  			var guids []string
   337  
   338  			BeforeEach(func() {
   339  				var apps []resources.Application
   340  
   341  				for i := 0; i < batcher.BatchSize*batches; i++ {
   342  					guids = append(guids, fmt.Sprintf("app-%d-guid", i))
   343  					apps = append(apps, resources.Application{
   344  						Name: fmt.Sprintf("app-%d-name", i),
   345  						GUID: fmt.Sprintf("app-%d-guid", i),
   346  					})
   347  				}
   348  
   349  				for b := 0; b < batches; b++ {
   350  					fakeCloudControllerClient.GetApplicationsReturnsOnCall(
   351  						b,
   352  						apps[:batcher.BatchSize],
   353  						ccv3.Warnings{"some-warning"},
   354  						nil,
   355  					)
   356  					apps = apps[batcher.BatchSize:]
   357  				}
   358  			})
   359  
   360  			It("makes many calls", func() {
   361  				apps, warnings, err := actor.GetApplicationsByGUIDs(guids)
   362  				Expect(len(apps)).To(Equal(batches * batcher.BatchSize))
   363  				Expect(apps).To(HaveLen(batcher.BatchSize * 10))
   364  				Expect(warnings).To(HaveLen(10))
   365  				Expect(err).NotTo(HaveOccurred())
   366  
   367  				Expect(fakeCloudControllerClient.GetApplicationsCallCount()).To(Equal(10))
   368  				Expect(fakeCloudControllerClient.GetApplicationsArgsForCall(0)).
   369  					NotTo(Equal(fakeCloudControllerClient.GetApplicationsArgsForCall(1)))
   370  			})
   371  		})
   372  	})
   373  
   374  	Describe("GetApplicationsByNameAndSpace", func() {
   375  		When("all of the requested apps exist", func() {
   376  			BeforeEach(func() {
   377  				fakeCloudControllerClient.GetApplicationsReturns(
   378  					[]resources.Application{
   379  						{
   380  							Name: "some-app-name",
   381  							GUID: "some-app-guid",
   382  						},
   383  						{
   384  							Name: "other-app-name",
   385  							GUID: "other-app-guid",
   386  						},
   387  					},
   388  					ccv3.Warnings{"some-warning"},
   389  					nil,
   390  				)
   391  			})
   392  
   393  			It("returns the applications and warnings", func() {
   394  				apps, warnings, err := actor.GetApplicationsByNamesAndSpace([]string{"some-app-name", "other-app-name"}, "some-space-guid")
   395  				Expect(err).ToNot(HaveOccurred())
   396  				Expect(apps).To(ConsistOf(
   397  					resources.Application{
   398  						Name: "some-app-name",
   399  						GUID: "some-app-guid",
   400  					},
   401  					resources.Application{
   402  						Name: "other-app-name",
   403  						GUID: "other-app-guid",
   404  					},
   405  				))
   406  				Expect(warnings).To(ConsistOf("some-warning"))
   407  
   408  				Expect(fakeCloudControllerClient.GetApplicationsCallCount()).To(Equal(1))
   409  				Expect(fakeCloudControllerClient.GetApplicationsArgsForCall(0)).To(ConsistOf(
   410  					ccv3.Query{Key: ccv3.NameFilter, Values: []string{"some-app-name", "other-app-name"}},
   411  					ccv3.Query{Key: ccv3.SpaceGUIDFilter, Values: []string{"some-space-guid"}},
   412  				))
   413  			})
   414  		})
   415  
   416  		When("at least one of the requested apps does not exist", func() {
   417  			BeforeEach(func() {
   418  				fakeCloudControllerClient.GetApplicationsReturns(
   419  					[]resources.Application{
   420  						{
   421  							Name: "some-app-name",
   422  						},
   423  					},
   424  					ccv3.Warnings{"some-warning"},
   425  					nil,
   426  				)
   427  			})
   428  
   429  			It("returns an ApplicationNotFoundError and the warnings", func() {
   430  				_, warnings, err := actor.GetApplicationsByNamesAndSpace([]string{"some-app-name", "other-app-name"}, "some-space-guid")
   431  				Expect(warnings).To(ConsistOf("some-warning"))
   432  				Expect(err).To(MatchError(actionerror.ApplicationsNotFoundError{}))
   433  			})
   434  		})
   435  
   436  		When("a given app has two routes", func() {
   437  			BeforeEach(func() {
   438  				fakeCloudControllerClient.GetApplicationsReturns(
   439  					[]resources.Application{
   440  						{
   441  							Name: "some-app-name",
   442  						},
   443  					},
   444  					ccv3.Warnings{"some-warning"},
   445  					nil,
   446  				)
   447  			})
   448  
   449  			It("returns an ApplicationNotFoundError and the warnings", func() {
   450  				_, warnings, err := actor.GetApplicationsByNamesAndSpace([]string{"some-app-name", "some-app-name"}, "some-space-guid")
   451  				Expect(err).ToNot(HaveOccurred())
   452  				Expect(warnings).To(ConsistOf("some-warning"))
   453  			})
   454  		})
   455  
   456  		When("the cloud controller client returns an error", func() {
   457  			var expectedError error
   458  
   459  			BeforeEach(func() {
   460  				expectedError = errors.New("I am a CloudControllerClient Error")
   461  				fakeCloudControllerClient.GetApplicationsReturns(
   462  					[]resources.Application{},
   463  					ccv3.Warnings{"some-warning"},
   464  					expectedError)
   465  			})
   466  
   467  			It("returns the warnings and the error", func() {
   468  				_, warnings, err := actor.GetApplicationsByNamesAndSpace([]string{"some-app-name"}, "some-space-guid")
   469  				Expect(warnings).To(ConsistOf("some-warning"))
   470  				Expect(err).To(MatchError(expectedError))
   471  			})
   472  		})
   473  	})
   474  
   475  	Describe("GetApplicationByNameAndSpace", func() {
   476  		When("the app exists", func() {
   477  			BeforeEach(func() {
   478  				fakeCloudControllerClient.GetApplicationsReturns(
   479  					[]resources.Application{
   480  						{
   481  							Name: "some-app-name",
   482  							GUID: "some-app-guid",
   483  							Metadata: &resources.Metadata{
   484  								Labels: map[string]types.NullString{
   485  									"some-key": types.NewNullString("some-value"),
   486  								},
   487  							},
   488  						},
   489  					},
   490  					ccv3.Warnings{"some-warning"},
   491  					nil,
   492  				)
   493  			})
   494  
   495  			It("returns the application and warnings", func() {
   496  				app, warnings, err := actor.GetApplicationByNameAndSpace("some-app-name", "some-space-guid")
   497  				Expect(err).ToNot(HaveOccurred())
   498  				Expect(app).To(Equal(resources.Application{
   499  					Name: "some-app-name",
   500  					GUID: "some-app-guid",
   501  					Metadata: &resources.Metadata{
   502  						Labels: map[string]types.NullString{"some-key": types.NewNullString("some-value")},
   503  					},
   504  				}))
   505  				Expect(warnings).To(ConsistOf("some-warning"))
   506  
   507  				Expect(fakeCloudControllerClient.GetApplicationsCallCount()).To(Equal(1))
   508  				Expect(fakeCloudControllerClient.GetApplicationsArgsForCall(0)).To(ConsistOf(
   509  					ccv3.Query{Key: ccv3.NameFilter, Values: []string{"some-app-name"}},
   510  					ccv3.Query{Key: ccv3.SpaceGUIDFilter, Values: []string{"some-space-guid"}},
   511  				))
   512  			})
   513  		})
   514  
   515  		When("the cloud controller client returns an error", func() {
   516  			var expectedError error
   517  
   518  			BeforeEach(func() {
   519  				expectedError = errors.New("I am a CloudControllerClient Error")
   520  				fakeCloudControllerClient.GetApplicationsReturns(
   521  					[]resources.Application{},
   522  					ccv3.Warnings{"some-warning"},
   523  					expectedError)
   524  			})
   525  
   526  			It("returns the warnings and the error", func() {
   527  				_, warnings, err := actor.GetApplicationByNameAndSpace("some-app-name", "some-space-guid")
   528  				Expect(warnings).To(ConsistOf("some-warning"))
   529  				Expect(err).To(MatchError(expectedError))
   530  			})
   531  		})
   532  
   533  		When("the app does not exist", func() {
   534  			BeforeEach(func() {
   535  				fakeCloudControllerClient.GetApplicationsReturns(
   536  					[]resources.Application{},
   537  					ccv3.Warnings{"some-warning"},
   538  					nil,
   539  				)
   540  			})
   541  
   542  			It("returns an ApplicationNotFoundError and the warnings", func() {
   543  				_, warnings, err := actor.GetApplicationByNameAndSpace("some-app-name", "some-space-guid")
   544  				Expect(warnings).To(ConsistOf("some-warning"))
   545  				Expect(err).To(MatchError(actionerror.ApplicationNotFoundError{Name: "some-app-name"}))
   546  			})
   547  		})
   548  	})
   549  
   550  	Describe("GetApplicationsBySpace", func() {
   551  		When("the there are applications in the space", func() {
   552  			BeforeEach(func() {
   553  				fakeCloudControllerClient.GetApplicationsReturns(
   554  					[]resources.Application{
   555  						{
   556  							GUID: "some-app-guid-1",
   557  							Name: "some-app-1",
   558  						},
   559  						{
   560  							GUID: "some-app-guid-2",
   561  							Name: "some-app-2",
   562  						},
   563  					},
   564  					ccv3.Warnings{"warning-1", "warning-2"},
   565  					nil,
   566  				)
   567  			})
   568  
   569  			It("returns the application and warnings", func() {
   570  				apps, warnings, err := actor.GetApplicationsBySpace("some-space-guid")
   571  				Expect(err).ToNot(HaveOccurred())
   572  				Expect(apps).To(ConsistOf(
   573  					resources.Application{
   574  						GUID: "some-app-guid-1",
   575  						Name: "some-app-1",
   576  					},
   577  					resources.Application{
   578  						GUID: "some-app-guid-2",
   579  						Name: "some-app-2",
   580  					},
   581  				))
   582  				Expect(warnings).To(ConsistOf("warning-1", "warning-2"))
   583  
   584  				Expect(fakeCloudControllerClient.GetApplicationsCallCount()).To(Equal(1))
   585  				Expect(fakeCloudControllerClient.GetApplicationsArgsForCall(0)).To(ConsistOf(
   586  					ccv3.Query{Key: ccv3.SpaceGUIDFilter, Values: []string{"some-space-guid"}},
   587  				))
   588  			})
   589  		})
   590  
   591  		When("the cloud controller client returns an error", func() {
   592  			var expectedError error
   593  
   594  			BeforeEach(func() {
   595  				expectedError = errors.New("I am a CloudControllerClient Error")
   596  				fakeCloudControllerClient.GetApplicationsReturns(
   597  					[]resources.Application{},
   598  					ccv3.Warnings{"some-warning"},
   599  					expectedError)
   600  			})
   601  
   602  			It("returns the error and warnings", func() {
   603  				_, warnings, err := actor.GetApplicationsBySpace("some-space-guid")
   604  				Expect(warnings).To(ConsistOf("some-warning"))
   605  				Expect(err).To(MatchError(expectedError))
   606  			})
   607  		})
   608  	})
   609  
   610  	Describe("CreateApplicationInSpace", func() {
   611  		var (
   612  			application resources.Application
   613  			warnings    Warnings
   614  			err         error
   615  		)
   616  
   617  		JustBeforeEach(func() {
   618  			application, warnings, err = actor.CreateApplicationInSpace(resources.Application{
   619  				Name:                "some-app-name",
   620  				LifecycleType:       constant.AppLifecycleTypeBuildpack,
   621  				LifecycleBuildpacks: []string{"buildpack-1", "buildpack-2"},
   622  			}, "some-space-guid")
   623  		})
   624  
   625  		When("the app successfully gets created", func() {
   626  			BeforeEach(func() {
   627  				fakeCloudControllerClient.CreateApplicationReturns(
   628  					resources.Application{
   629  						Name:                "some-app-name",
   630  						GUID:                "some-app-guid",
   631  						LifecycleType:       constant.AppLifecycleTypeBuildpack,
   632  						LifecycleBuildpacks: []string{"buildpack-1", "buildpack-2"},
   633  					},
   634  					ccv3.Warnings{"some-warning"},
   635  					nil,
   636  				)
   637  			})
   638  
   639  			It("creates and returns the application and warnings", func() {
   640  				Expect(err).ToNot(HaveOccurred())
   641  				Expect(application).To(Equal(resources.Application{
   642  					Name:                "some-app-name",
   643  					GUID:                "some-app-guid",
   644  					LifecycleType:       constant.AppLifecycleTypeBuildpack,
   645  					LifecycleBuildpacks: []string{"buildpack-1", "buildpack-2"},
   646  				}))
   647  				Expect(warnings).To(ConsistOf("some-warning"))
   648  
   649  				Expect(fakeCloudControllerClient.CreateApplicationCallCount()).To(Equal(1))
   650  				Expect(fakeCloudControllerClient.CreateApplicationArgsForCall(0)).To(Equal(resources.Application{
   651  					Name:                "some-app-name",
   652  					SpaceGUID:           "some-space-guid",
   653  					LifecycleType:       constant.AppLifecycleTypeBuildpack,
   654  					LifecycleBuildpacks: []string{"buildpack-1", "buildpack-2"},
   655  				}))
   656  			})
   657  		})
   658  
   659  		When("the cc client returns an error", func() {
   660  			var expectedError error
   661  
   662  			BeforeEach(func() {
   663  				expectedError = errors.New("I am a CloudControllerClient Error")
   664  				fakeCloudControllerClient.CreateApplicationReturns(
   665  					resources.Application{},
   666  					ccv3.Warnings{"some-warning"},
   667  					expectedError,
   668  				)
   669  			})
   670  
   671  			It("raises the error and warnings", func() {
   672  				Expect(err).To(MatchError(expectedError))
   673  				Expect(warnings).To(ConsistOf("some-warning"))
   674  			})
   675  		})
   676  
   677  		When("the cc client returns an NameNotUniqueInSpaceError", func() {
   678  			BeforeEach(func() {
   679  				fakeCloudControllerClient.CreateApplicationReturns(
   680  					resources.Application{},
   681  					ccv3.Warnings{"some-warning"},
   682  					ccerror.NameNotUniqueInSpaceError{},
   683  				)
   684  			})
   685  
   686  			It("returns the NameNotUniqueInSpaceError and warnings", func() {
   687  				Expect(err).To(MatchError(ccerror.NameNotUniqueInSpaceError{}))
   688  				Expect(warnings).To(ConsistOf("some-warning"))
   689  			})
   690  		})
   691  	})
   692  
   693  	Describe("UpdateApplication", func() {
   694  		var (
   695  			submitApp, resultApp resources.Application
   696  			warnings             Warnings
   697  			err                  error
   698  		)
   699  
   700  		JustBeforeEach(func() {
   701  			submitApp = resources.Application{
   702  				GUID:                "some-app-guid",
   703  				StackName:           "some-stack-name",
   704  				Name:                "some-app-name",
   705  				LifecycleType:       constant.AppLifecycleTypeBuildpack,
   706  				LifecycleBuildpacks: []string{"buildpack-1", "buildpack-2"},
   707  				Metadata: &resources.Metadata{Labels: map[string]types.NullString{
   708  					"some-label":  types.NewNullString("some-value"),
   709  					"other-label": types.NewNullString("other-value"),
   710  				}},
   711  			}
   712  
   713  			resultApp, warnings, err = actor.UpdateApplication(submitApp)
   714  		})
   715  
   716  		When("the app successfully gets updated", func() {
   717  			var apiResponseApp resources.Application
   718  
   719  			BeforeEach(func() {
   720  				apiResponseApp = resources.Application{
   721  					GUID:                "response-app-guid",
   722  					StackName:           "response-stack-name",
   723  					Name:                "response-app-name",
   724  					LifecycleType:       constant.AppLifecycleTypeBuildpack,
   725  					LifecycleBuildpacks: []string{"response-buildpack-1", "response-buildpack-2"},
   726  				}
   727  				fakeCloudControllerClient.UpdateApplicationReturns(
   728  					apiResponseApp,
   729  					ccv3.Warnings{"some-warning"},
   730  					nil,
   731  				)
   732  			})
   733  
   734  			It("creates and returns the application and warnings", func() {
   735  				Expect(err).ToNot(HaveOccurred())
   736  				Expect(resultApp).To(Equal(resources.Application{
   737  					Name:                apiResponseApp.Name,
   738  					GUID:                apiResponseApp.GUID,
   739  					StackName:           apiResponseApp.StackName,
   740  					LifecycleType:       apiResponseApp.LifecycleType,
   741  					LifecycleBuildpacks: apiResponseApp.LifecycleBuildpacks,
   742  				}))
   743  				Expect(warnings).To(ConsistOf("some-warning"))
   744  
   745  				Expect(fakeCloudControllerClient.UpdateApplicationCallCount()).To(Equal(1))
   746  				Expect(fakeCloudControllerClient.UpdateApplicationArgsForCall(0)).To(Equal(resources.Application{
   747  					GUID:                submitApp.GUID,
   748  					StackName:           submitApp.StackName,
   749  					LifecycleType:       submitApp.LifecycleType,
   750  					LifecycleBuildpacks: submitApp.LifecycleBuildpacks,
   751  					Name:                submitApp.Name,
   752  					Metadata:            submitApp.Metadata,
   753  				}))
   754  			})
   755  		})
   756  
   757  		When("the cc client returns an error", func() {
   758  			var expectedError error
   759  
   760  			BeforeEach(func() {
   761  				expectedError = errors.New("I am a CloudControllerClient Error")
   762  				fakeCloudControllerClient.UpdateApplicationReturns(
   763  					resources.Application{},
   764  					ccv3.Warnings{"some-warning"},
   765  					expectedError,
   766  				)
   767  			})
   768  
   769  			It("raises the error and warnings", func() {
   770  				Expect(err).To(MatchError(expectedError))
   771  				Expect(warnings).To(ConsistOf("some-warning"))
   772  			})
   773  		})
   774  	})
   775  
   776  	Describe("UpdateApplicationName", func() {
   777  		var (
   778  			resultApp           resources.Application
   779  			newAppName, appGUID string
   780  			warnings            Warnings
   781  			err                 error
   782  		)
   783  
   784  		JustBeforeEach(func() {
   785  			newAppName = "some-new-app-name"
   786  			appGUID = "some-app-guid"
   787  
   788  			resultApp, warnings, err = actor.UpdateApplicationName(newAppName, appGUID)
   789  		})
   790  
   791  		When("the app successfully gets updated", func() {
   792  			var apiResponseApp resources.Application
   793  
   794  			BeforeEach(func() {
   795  				apiResponseApp = resources.Application{
   796  					GUID:                "response-app-guid",
   797  					StackName:           "response-stack-name",
   798  					Name:                "response-app-name",
   799  					LifecycleType:       constant.AppLifecycleTypeBuildpack,
   800  					LifecycleBuildpacks: []string{"response-buildpack-1", "response-buildpack-2"},
   801  				}
   802  				fakeCloudControllerClient.UpdateApplicationNameReturns(
   803  					apiResponseApp,
   804  					ccv3.Warnings{"some-warning"},
   805  					nil,
   806  				)
   807  			})
   808  
   809  			It("creates and returns the application and warnings", func() {
   810  				Expect(err).ToNot(HaveOccurred())
   811  				Expect(resultApp).To(Equal(resources.Application{
   812  					Name:                apiResponseApp.Name,
   813  					GUID:                apiResponseApp.GUID,
   814  					StackName:           apiResponseApp.StackName,
   815  					LifecycleType:       apiResponseApp.LifecycleType,
   816  					LifecycleBuildpacks: apiResponseApp.LifecycleBuildpacks,
   817  				}))
   818  				Expect(warnings).To(ConsistOf("some-warning"))
   819  
   820  				Expect(fakeCloudControllerClient.UpdateApplicationNameCallCount()).To(Equal(1))
   821  				appName, appGuid := fakeCloudControllerClient.UpdateApplicationNameArgsForCall(0)
   822  				Expect(appName).To(Equal("some-new-app-name"))
   823  				Expect(appGuid).To(Equal("some-app-guid"))
   824  			})
   825  		})
   826  
   827  		When("the cc client returns an error", func() {
   828  			var expectedError error
   829  
   830  			BeforeEach(func() {
   831  				expectedError = errors.New("I am a CloudControllerClient Error")
   832  				fakeCloudControllerClient.UpdateApplicationNameReturns(
   833  					resources.Application{},
   834  					ccv3.Warnings{"some-warning"},
   835  					expectedError,
   836  				)
   837  			})
   838  
   839  			It("raises the error and warnings", func() {
   840  				Expect(err).To(MatchError(expectedError))
   841  				Expect(warnings).To(ConsistOf("some-warning"))
   842  			})
   843  		})
   844  	})
   845  
   846  	Describe("PollStart", func() {
   847  		var (
   848  			app                   resources.Application
   849  			noWait                bool
   850  			handleInstanceDetails func(string)
   851  
   852  			done chan bool
   853  
   854  			warnings                Warnings
   855  			executeErr              error
   856  			reportedInstanceDetails []string
   857  		)
   858  
   859  		BeforeEach(func() {
   860  			done = make(chan bool)
   861  			fakeConfig.StartupTimeoutReturns(2 * time.Second)
   862  			fakeConfig.PollingIntervalReturns(1 * time.Second)
   863  			app = resources.Application{GUID: "some-guid"}
   864  			noWait = false
   865  
   866  			reportedInstanceDetails = []string{}
   867  			handleInstanceDetails = func(instanceDetails string) {
   868  				reportedInstanceDetails = append(reportedInstanceDetails, instanceDetails)
   869  			}
   870  		})
   871  
   872  		JustBeforeEach(func() {
   873  			go func() {
   874  				defer close(done)
   875  				warnings, executeErr = actor.PollStart(app, noWait, handleInstanceDetails)
   876  				done <- true
   877  			}()
   878  		})
   879  
   880  		It("gets the apps processes", func() {
   881  			// advanced clock so function exits
   882  			fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
   883  
   884  			// wait for function to finish
   885  			Eventually(done).Should(Receive(BeTrue()))
   886  
   887  			Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(1))
   888  			Expect(fakeCloudControllerClient.GetApplicationProcessesArgsForCall(0)).To(Equal("some-guid"))
   889  
   890  		})
   891  
   892  		When("getting the application processes fails", func() {
   893  			BeforeEach(func() {
   894  				fakeCloudControllerClient.GetApplicationProcessesReturns(nil, ccv3.Warnings{"get-app-warning-1", "get-app-warning-2"}, errors.New("some-error"))
   895  			})
   896  
   897  			It("returns the error and all warnings", func() {
   898  				// wait for function to finish
   899  				Eventually(done).Should(Receive(BeTrue()))
   900  
   901  				Expect(executeErr).To(MatchError(errors.New("some-error")))
   902  				Expect(warnings).To(ConsistOf("get-app-warning-1", "get-app-warning-2"))
   903  			})
   904  		})
   905  
   906  		When("getting the application process succeeds", func() {
   907  			BeforeEach(func() {
   908  				fakeCloudControllerClient.GetApplicationProcessesReturns(
   909  					[]resources.Process{
   910  						{GUID: "process1", Type: "web"},
   911  					},
   912  					ccv3.Warnings{"get-app-warning-1"},
   913  					nil,
   914  				)
   915  
   916  			})
   917  
   918  			It("gets the startup timeout", func() {
   919  				// advanced clock so function exits
   920  				fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
   921  
   922  				// wait for function to finish
   923  				Eventually(done).Should(Receive(BeTrue()))
   924  
   925  				Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
   926  			})
   927  
   928  			When("the no-wait flag is provided", func() {
   929  				BeforeEach(func() {
   930  					noWait = true
   931  					fakeCloudControllerClient.GetApplicationProcessesReturns(
   932  						[]resources.Process{
   933  							{GUID: "process1", Type: "web"},
   934  							{GUID: "process2", Type: "worker"},
   935  						},
   936  						ccv3.Warnings{"get-app-warning-1"},
   937  						nil,
   938  					)
   939  				})
   940  
   941  				It("filters out the non web processes", func() {
   942  					// send something on the timer channel
   943  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
   944  
   945  					// Wait for function to finish
   946  					Eventually(done).Should(Receive(BeTrue()))
   947  
   948  					// assert on the cc call made within poll processes to make sure there is only the web process
   949  					Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(1))
   950  					Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process1"))
   951  
   952  				})
   953  			})
   954  
   955  			When("polling processes returns an error", func() {
   956  				BeforeEach(func() {
   957  					fakeCloudControllerClient.GetProcessInstancesReturns(nil, ccv3.Warnings{"poll-process-warning"}, errors.New("poll-process-error"))
   958  				})
   959  
   960  				It("returns the error and warnings", func() {
   961  					// send something on the timer channel
   962  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
   963  
   964  					// Wait for function to finish
   965  					Eventually(done).Should(Receive(BeTrue()))
   966  
   967  					Expect(executeErr).Should(MatchError("poll-process-error"))
   968  					Expect(warnings).Should(ConsistOf("poll-process-warning", "get-app-warning-1"))
   969  				})
   970  			})
   971  
   972  			When("polling start times out", func() {
   973  				BeforeEach(func() {
   974  					fakeCloudControllerClient.GetProcessInstancesReturns(
   975  						[]ccv3.ProcessInstance{
   976  							{State: constant.ProcessInstanceStarting},
   977  						},
   978  						ccv3.Warnings{"poll-process-warning"},
   979  						nil,
   980  					)
   981  
   982  					fakeConfig.StartupTimeoutReturns(2 * time.Millisecond)
   983  				})
   984  
   985  				It("returns a timeout error and any warnings", func() {
   986  					// send something on the timer channel for first tick
   987  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
   988  
   989  					fakeClock.Increment(1 * time.Millisecond)
   990  
   991  					// Wait for function to finish
   992  					Eventually(done).Should(Receive(BeTrue()))
   993  
   994  					Expect(executeErr).To(MatchError(actionerror.StartupTimeoutError{}))
   995  					Expect(warnings).To(ConsistOf("poll-process-warning", "get-app-warning-1"))
   996  				})
   997  			})
   998  
   999  			When("polling process eventually returns we should stop polling", func() {
  1000  				BeforeEach(func() {
  1001  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(0,
  1002  						[]ccv3.ProcessInstance{
  1003  							{State: constant.ProcessInstanceStarting},
  1004  						},
  1005  						ccv3.Warnings{"poll-process-warning1"},
  1006  						nil,
  1007  					)
  1008  
  1009  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(1,
  1010  						[]ccv3.ProcessInstance{
  1011  							{State: constant.ProcessInstanceRunning},
  1012  						},
  1013  						ccv3.Warnings{"poll-process-warning2"},
  1014  						nil,
  1015  					)
  1016  				})
  1017  
  1018  				It("returns success and any warnings", func() {
  1019  					// send something on the timer channel
  1020  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1021  
  1022  					Eventually(fakeConfig.PollingIntervalCallCount).Should(Equal(1))
  1023  
  1024  					fakeClock.Increment(1 * time.Second)
  1025  
  1026  					// Wait for function to finish
  1027  					Eventually(done).Should(Receive(BeTrue()))
  1028  					Expect(executeErr).NotTo(HaveOccurred())
  1029  					Expect(warnings).To(ConsistOf("poll-process-warning1", "get-app-warning-1", "poll-process-warning2"))
  1030  				})
  1031  
  1032  			})
  1033  		})
  1034  	})
  1035  
  1036  	Describe("PollStartForRolling", func() {
  1037  		var (
  1038  			app                   resources.Application
  1039  			deploymentGUID        string
  1040  			noWait                bool
  1041  			handleInstanceDetails func(string)
  1042  
  1043  			done chan bool
  1044  
  1045  			warnings                Warnings
  1046  			executeErr              error
  1047  			reportedInstanceDetails []string
  1048  		)
  1049  
  1050  		BeforeEach(func() {
  1051  			reportedInstanceDetails = []string{}
  1052  			handleInstanceDetails = func(instanceDetails string) {
  1053  				reportedInstanceDetails = append(reportedInstanceDetails, instanceDetails)
  1054  			}
  1055  
  1056  			app = resources.Application{GUID: "some-rolling-app-guid"}
  1057  			deploymentGUID = "some-deployment-guid"
  1058  			noWait = false
  1059  
  1060  			done = make(chan bool)
  1061  
  1062  			fakeConfig.StartupTimeoutReturns(5 * time.Second)
  1063  			fakeConfig.PollingIntervalReturns(1 * time.Second)
  1064  		})
  1065  
  1066  		JustBeforeEach(func() {
  1067  			go func() {
  1068  				warnings, executeErr = actor.PollStartForRolling(app, deploymentGUID, noWait, handleInstanceDetails)
  1069  				done <- true
  1070  			}()
  1071  		})
  1072  
  1073  		When("There is a non-timeout failure in the loop", func() {
  1074  			// this may need to be expanded to also include when the deployment is superseded or cancelled
  1075  			When("getting the deployment fails", func() {
  1076  				When("it is because the deployment was cancelled", func() {
  1077  					BeforeEach(func() {
  1078  						fakeCloudControllerClient.GetDeploymentReturns(
  1079  							resources.Deployment{
  1080  								StatusValue:  constant.DeploymentStatusValueFinalized,
  1081  								StatusReason: constant.DeploymentStatusReasonCanceled,
  1082  							},
  1083  							ccv3.Warnings{"get-deployment-warning"},
  1084  							nil,
  1085  						)
  1086  					})
  1087  
  1088  					It("returns warnings and the error", func() {
  1089  						// initial tick
  1090  						fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1091  
  1092  						// wait for func to finish
  1093  						Eventually(done).Should(Receive(BeTrue()))
  1094  
  1095  						Expect(executeErr).To(MatchError("Deployment has been canceled"))
  1096  						Expect(warnings).To(ConsistOf("get-deployment-warning"))
  1097  
  1098  						Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(1))
  1099  						Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1100  
  1101  						Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(0))
  1102  						Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(0))
  1103  
  1104  						Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
  1105  					})
  1106  
  1107  				})
  1108  
  1109  				When("it is because the deployment was superseded", func() {
  1110  					BeforeEach(func() {
  1111  						fakeCloudControllerClient.GetDeploymentReturns(
  1112  							resources.Deployment{
  1113  								StatusValue:  constant.DeploymentStatusValueFinalized,
  1114  								StatusReason: constant.DeploymentStatusReasonSuperseded,
  1115  							},
  1116  							ccv3.Warnings{"get-deployment-warning"},
  1117  							nil,
  1118  						)
  1119  					})
  1120  
  1121  					It("returns warnings and the error", func() {
  1122  						// initial tick
  1123  						fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1124  
  1125  						// wait for func to finish
  1126  						Eventually(done).Should(Receive(BeTrue()))
  1127  
  1128  						Expect(executeErr).To(MatchError("Deployment has been superseded"))
  1129  						Expect(warnings).To(ConsistOf("get-deployment-warning"))
  1130  
  1131  						Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(1))
  1132  						Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1133  
  1134  						Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(0))
  1135  						Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(0))
  1136  
  1137  						Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
  1138  					})
  1139  
  1140  				})
  1141  
  1142  				When("it is because of an API error", func() {
  1143  					BeforeEach(func() {
  1144  						fakeCloudControllerClient.GetDeploymentReturns(
  1145  							resources.Deployment{},
  1146  							ccv3.Warnings{"get-deployment-warning"},
  1147  							errors.New("get-deployment-error"),
  1148  						)
  1149  					})
  1150  
  1151  					It("returns warnings and the error", func() {
  1152  						// initial tick
  1153  						fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1154  
  1155  						// wait for func to finish
  1156  						Eventually(done).Should(Receive(BeTrue()))
  1157  
  1158  						Expect(executeErr).To(MatchError("get-deployment-error"))
  1159  						Expect(warnings).To(ConsistOf("get-deployment-warning"))
  1160  
  1161  						Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(1))
  1162  						Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1163  
  1164  						Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(0))
  1165  						Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(0))
  1166  
  1167  						Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
  1168  					})
  1169  
  1170  				})
  1171  			})
  1172  
  1173  			When("getting the deployment succeeds", func() {
  1174  				BeforeEach(func() {
  1175  					// get processes requires the deployment to be deployed so we need this to indirectly test the error case
  1176  					fakeCloudControllerClient.GetDeploymentReturns(
  1177  						resources.Deployment{StatusValue: constant.DeploymentStatusValueFinalized, StatusReason: constant.DeploymentStatusReasonDeployed},
  1178  						ccv3.Warnings{"get-deployment-warning"},
  1179  						nil,
  1180  					)
  1181  
  1182  				})
  1183  
  1184  				When("getting the processes fails", func() {
  1185  					BeforeEach(func() {
  1186  						fakeCloudControllerClient.GetApplicationProcessesReturns(
  1187  							[]resources.Process{},
  1188  							ccv3.Warnings{"get-processes-warning"},
  1189  							errors.New("get-processes-error"),
  1190  						)
  1191  					})
  1192  
  1193  					It("returns warnings and the error", func() {
  1194  						// initial tick
  1195  						fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1196  
  1197  						// wait for func to finish
  1198  						Eventually(done).Should(Receive(BeTrue()))
  1199  
  1200  						Expect(executeErr).To(MatchError("get-processes-error"))
  1201  						Expect(warnings).To(ConsistOf("get-deployment-warning", "get-processes-warning"))
  1202  
  1203  						Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(1))
  1204  						Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1205  
  1206  						Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(1))
  1207  						Expect(fakeCloudControllerClient.GetApplicationProcessesArgsForCall(0)).To(Equal(app.GUID))
  1208  
  1209  						Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(0))
  1210  
  1211  					})
  1212  				})
  1213  
  1214  				When("getting the processes succeeds", func() {
  1215  					BeforeEach(func() {
  1216  						fakeCloudControllerClient.GetApplicationProcessesReturns(
  1217  							[]resources.Process{{GUID: "process-guid"}},
  1218  							ccv3.Warnings{"get-processes-warning"},
  1219  							nil,
  1220  						)
  1221  					})
  1222  
  1223  					When("polling the processes fails", func() {
  1224  						BeforeEach(func() {
  1225  							fakeCloudControllerClient.GetProcessInstancesReturns(
  1226  								[]ccv3.ProcessInstance{},
  1227  								ccv3.Warnings{"poll-processes-warning"},
  1228  								errors.New("poll-processes-error"),
  1229  							)
  1230  						})
  1231  
  1232  						It("returns all warnings and errors", func() {
  1233  							// initial tick
  1234  							fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1235  
  1236  							// wait for func to finish
  1237  							Eventually(done).Should(Receive(BeTrue()))
  1238  
  1239  							Expect(executeErr).To(MatchError("poll-processes-error"))
  1240  							Expect(warnings).To(ConsistOf("get-deployment-warning", "get-processes-warning", "poll-processes-warning"))
  1241  
  1242  							Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(1))
  1243  							Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1244  
  1245  							Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(1))
  1246  							Expect(fakeCloudControllerClient.GetApplicationProcessesArgsForCall(0)).To(Equal(app.GUID))
  1247  
  1248  							Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(1))
  1249  							Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process-guid"))
  1250  						})
  1251  
  1252  					})
  1253  				})
  1254  
  1255  			})
  1256  
  1257  		})
  1258  
  1259  		// intentionally ignore the no-wait flag here for simplicity. One of these two things must cause timeout regardless of no-wait state
  1260  		When("there is a timeout error", func() {
  1261  			BeforeEach(func() {
  1262  				// 1 millisecond for initial tick then 1 to trigger timeout
  1263  				fakeConfig.StartupTimeoutReturns(2 * time.Millisecond)
  1264  			})
  1265  
  1266  			When("the deployment never deploys", func() {
  1267  				BeforeEach(func() {
  1268  					fakeCloudControllerClient.GetDeploymentReturns(
  1269  						resources.Deployment{StatusValue: constant.DeploymentStatusValueActive},
  1270  						ccv3.Warnings{"get-deployment-warning"},
  1271  						nil,
  1272  					)
  1273  					fakeCloudControllerClient.CancelDeploymentReturns(
  1274  						ccv3.Warnings{"cancel-deployment-warning"},
  1275  						nil,
  1276  					)
  1277  				})
  1278  
  1279  				It("returns a timeout error and any warnings and cancels the deployment", func() {
  1280  					// initial tick
  1281  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1282  
  1283  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(1))
  1284  
  1285  					// timeout tick
  1286  					fakeClock.Increment(1 * time.Millisecond)
  1287  
  1288  					Eventually(fakeCloudControllerClient.CancelDeploymentCallCount).Should(Equal(1))
  1289  
  1290  					// wait for func to finish
  1291  					Eventually(done).Should(Receive(BeTrue()))
  1292  
  1293  					Expect(executeErr).To(MatchError(actionerror.StartupTimeoutError{}))
  1294  					Expect(warnings).To(ConsistOf("get-deployment-warning", "cancel-deployment-warning"))
  1295  				})
  1296  
  1297  				When("the cancel deployment fails", func() {
  1298  					BeforeEach(func() {
  1299  						fakeCloudControllerClient.CancelDeploymentReturns(
  1300  							ccv3.Warnings{"cancel-deployment-warning"},
  1301  							errors.New("cancel-deployment-error"),
  1302  						)
  1303  					})
  1304  
  1305  					It("returns a timeout error and any warnings and cancels the deployment", func() {
  1306  						// initial tick
  1307  						fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1308  
  1309  						Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(1))
  1310  
  1311  						// timeout tick
  1312  						fakeClock.Increment(1 * time.Millisecond)
  1313  
  1314  						Eventually(fakeCloudControllerClient.CancelDeploymentCallCount).Should(Equal(1))
  1315  
  1316  						// wait for func to finish
  1317  						Eventually(done).Should(Receive(BeTrue()))
  1318  
  1319  						Expect(executeErr).To(MatchError("cancel-deployment-error"))
  1320  						Expect(warnings).To(ConsistOf("get-deployment-warning", "cancel-deployment-warning"))
  1321  					})
  1322  
  1323  				})
  1324  
  1325  			})
  1326  
  1327  			When("the processes dont become healthy", func() {
  1328  				BeforeEach(func() {
  1329  					fakeCloudControllerClient.GetDeploymentReturns(
  1330  						resources.Deployment{StatusValue: constant.DeploymentStatusValueFinalized, StatusReason: constant.DeploymentStatusReasonDeployed},
  1331  						ccv3.Warnings{"get-deployment-warning"},
  1332  						nil,
  1333  					)
  1334  
  1335  					fakeCloudControllerClient.GetApplicationProcessesReturns(
  1336  						[]resources.Process{{GUID: "process-guid"}},
  1337  						ccv3.Warnings{"get-processes-warning"},
  1338  						nil,
  1339  					)
  1340  
  1341  					fakeCloudControllerClient.GetProcessInstancesReturns(
  1342  						[]ccv3.ProcessInstance{{State: constant.ProcessInstanceStarting}},
  1343  						ccv3.Warnings{"poll-processes-warning"},
  1344  						nil,
  1345  					)
  1346  				})
  1347  
  1348  				It("returns a timeout error and any warnings", func() {
  1349  					// initial tick
  1350  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1351  
  1352  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(1))
  1353  					Eventually(fakeCloudControllerClient.GetApplicationProcessesCallCount).Should(Equal(1))
  1354  					Eventually(fakeCloudControllerClient.GetProcessInstancesCallCount).Should(Equal(1))
  1355  
  1356  					// timeout tick
  1357  					fakeClock.Increment(1 * time.Millisecond)
  1358  
  1359  					// wait for func to finish
  1360  					Eventually(done).Should(Receive(BeTrue()))
  1361  
  1362  					Expect(executeErr).To(MatchError(actionerror.StartupTimeoutError{}))
  1363  					Expect(warnings).To(ConsistOf("get-deployment-warning", "get-processes-warning", "poll-processes-warning"))
  1364  				})
  1365  
  1366  			})
  1367  		})
  1368  
  1369  		When("things eventually become healthy", func() {
  1370  			When("the no wait flag is given", func() {
  1371  				BeforeEach(func() {
  1372  					// in total three loops 1: deployment still deploying 2: deployment deployed processes starting 3: processes started
  1373  					noWait = true
  1374  
  1375  					// Always return deploying as a way to check we respect no wait
  1376  					fakeCloudControllerClient.GetDeploymentReturns(
  1377  						resources.Deployment{
  1378  							StatusValue:  constant.DeploymentStatusValueActive,
  1379  							NewProcesses: []resources.Process{{GUID: "new-deployment-process"}},
  1380  						},
  1381  						ccv3.Warnings{"get-deployment-warning"},
  1382  						nil,
  1383  					)
  1384  
  1385  					// We only poll the processes. Two loops for fun
  1386  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(0,
  1387  						[]ccv3.ProcessInstance{{State: constant.ProcessInstanceStarting}},
  1388  						ccv3.Warnings{"poll-processes-warning-1"},
  1389  						nil,
  1390  					)
  1391  
  1392  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(1,
  1393  						[]ccv3.ProcessInstance{{State: constant.ProcessInstanceRunning}},
  1394  						ccv3.Warnings{"poll-processes-warning-2"},
  1395  						nil,
  1396  					)
  1397  				})
  1398  
  1399  				It("polls the start of the application correctly and returns warnings and no error", func() {
  1400  					// Initial tick
  1401  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1402  
  1403  					// assert one of our watcher is the timeout
  1404  					Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
  1405  
  1406  					// the first time through we always get the deployment regardless of no-wait
  1407  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(1))
  1408  					Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1409  					Eventually(fakeCloudControllerClient.GetProcessInstancesCallCount).Should(Equal(1))
  1410  					Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("new-deployment-process"))
  1411  					Eventually(fakeConfig.PollingIntervalCallCount).Should(Equal(1))
  1412  
  1413  					fakeClock.Increment(1 * time.Second)
  1414  
  1415  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(2))
  1416  					Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1417  					Eventually(fakeCloudControllerClient.GetProcessInstancesCallCount).Should(Equal(2))
  1418  					Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("new-deployment-process"))
  1419  
  1420  					Eventually(done).Should(Receive(BeTrue()))
  1421  
  1422  					Expect(executeErr).NotTo(HaveOccurred())
  1423  					Expect(warnings).To(ConsistOf(
  1424  						"get-deployment-warning",
  1425  						"poll-processes-warning-1",
  1426  						"get-deployment-warning",
  1427  						"poll-processes-warning-2",
  1428  					))
  1429  
  1430  					Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(2))
  1431  					Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(0))
  1432  					Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(2))
  1433  					Expect(fakeConfig.PollingIntervalCallCount()).To(Equal(1))
  1434  
  1435  				})
  1436  
  1437  			})
  1438  
  1439  			When("the no wait flag is not given", func() {
  1440  				BeforeEach(func() {
  1441  					// in total three loops 1: deployment still deploying 2: deployment deployed processes starting 3: processes started
  1442  					fakeCloudControllerClient.GetDeploymentReturnsOnCall(0,
  1443  						resources.Deployment{StatusValue: constant.DeploymentStatusValueActive},
  1444  						ccv3.Warnings{"get-deployment-warning-1"},
  1445  						nil,
  1446  					)
  1447  
  1448  					// Poll the deployment twice to make sure we are polling (one in the above before each)
  1449  					fakeCloudControllerClient.GetDeploymentReturnsOnCall(1,
  1450  						resources.Deployment{StatusValue: constant.DeploymentStatusValueFinalized, StatusReason: constant.DeploymentStatusReasonDeployed},
  1451  						ccv3.Warnings{"get-deployment-warning-2"},
  1452  						nil,
  1453  					)
  1454  
  1455  					// then we get the processes. This should only be called once
  1456  					fakeCloudControllerClient.GetApplicationProcessesReturns(
  1457  						[]resources.Process{{GUID: "process-guid"}},
  1458  						ccv3.Warnings{"get-processes-warning"},
  1459  						nil,
  1460  					)
  1461  
  1462  					// then we poll the processes. Two loops for fun
  1463  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(0,
  1464  						[]ccv3.ProcessInstance{{State: constant.ProcessInstanceStarting}},
  1465  						ccv3.Warnings{"poll-processes-warning-1"},
  1466  						nil,
  1467  					)
  1468  
  1469  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(1,
  1470  						[]ccv3.ProcessInstance{{State: constant.ProcessInstanceRunning}},
  1471  						ccv3.Warnings{"poll-processes-warning-2"},
  1472  						nil,
  1473  					)
  1474  				})
  1475  
  1476  				It("polls the start of the application correctly and returns warnings and no error", func() {
  1477  					// Initial tick
  1478  					fakeClock.WaitForNWatchersAndIncrement(1*time.Millisecond, 2)
  1479  
  1480  					// assert one of our watchers is for the timeout
  1481  					Expect(fakeConfig.StartupTimeoutCallCount()).To(Equal(1))
  1482  
  1483  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(1))
  1484  					Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(0)).To(Equal(deploymentGUID))
  1485  					Eventually(fakeConfig.PollingIntervalCallCount).Should(Equal(1))
  1486  
  1487  					// start the second loop where the deployment is deployed so we poll processes
  1488  					fakeClock.Increment(1 * time.Second)
  1489  
  1490  					Eventually(fakeCloudControllerClient.GetDeploymentCallCount).Should(Equal(2))
  1491  					Expect(fakeCloudControllerClient.GetDeploymentArgsForCall(1)).To(Equal(deploymentGUID))
  1492  					Eventually(fakeCloudControllerClient.GetApplicationProcessesCallCount).Should(Equal(1))
  1493  					Expect(fakeCloudControllerClient.GetApplicationProcessesArgsForCall(0)).To(Equal(app.GUID))
  1494  					Eventually(fakeCloudControllerClient.GetProcessInstancesCallCount).Should(Equal(1))
  1495  					Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process-guid"))
  1496  					Eventually(fakeConfig.PollingIntervalCallCount).Should(Equal(2))
  1497  
  1498  					fakeClock.Increment(1 * time.Second)
  1499  
  1500  					// we should stop polling because it is deployed
  1501  					Eventually(fakeCloudControllerClient.GetProcessInstancesCallCount).Should(Equal(2))
  1502  					Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process-guid"))
  1503  
  1504  					Eventually(done).Should(Receive(BeTrue()))
  1505  
  1506  					Expect(executeErr).NotTo(HaveOccurred())
  1507  					Expect(warnings).To(ConsistOf(
  1508  						"get-deployment-warning-1",
  1509  						"get-deployment-warning-2",
  1510  						"get-processes-warning",
  1511  						"poll-processes-warning-1",
  1512  						"poll-processes-warning-2",
  1513  					))
  1514  
  1515  					Expect(fakeCloudControllerClient.GetDeploymentCallCount()).To(Equal(2))
  1516  					Expect(fakeCloudControllerClient.GetApplicationProcessesCallCount()).To(Equal(1))
  1517  					Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(2))
  1518  					Expect(fakeConfig.PollingIntervalCallCount()).To(Equal(2))
  1519  
  1520  				})
  1521  
  1522  			})
  1523  
  1524  		})
  1525  	})
  1526  
  1527  	Describe("SetApplicationProcessHealthCheckTypeByNameAndSpace", func() {
  1528  		var (
  1529  			healthCheckType     constant.HealthCheckType
  1530  			healthCheckEndpoint string
  1531  
  1532  			warnings Warnings
  1533  			err      error
  1534  			app      resources.Application
  1535  		)
  1536  
  1537  		BeforeEach(func() {
  1538  			healthCheckType = constant.HTTP
  1539  			healthCheckEndpoint = "some-http-endpoint"
  1540  		})
  1541  
  1542  		JustBeforeEach(func() {
  1543  			app, warnings, err = actor.SetApplicationProcessHealthCheckTypeByNameAndSpace(
  1544  				"some-app-name",
  1545  				"some-space-guid",
  1546  				healthCheckType,
  1547  				healthCheckEndpoint,
  1548  				"some-process-type",
  1549  				42,
  1550  			)
  1551  		})
  1552  
  1553  		When("getting application returns an error", func() {
  1554  			var expectedErr error
  1555  
  1556  			BeforeEach(func() {
  1557  				expectedErr = errors.New("some-error")
  1558  				fakeCloudControllerClient.GetApplicationsReturns(
  1559  					[]resources.Application{},
  1560  					ccv3.Warnings{"some-warning"},
  1561  					expectedErr,
  1562  				)
  1563  			})
  1564  
  1565  			It("returns the error and warnings", func() {
  1566  				Expect(err).To(Equal(expectedErr))
  1567  				Expect(warnings).To(ConsistOf("some-warning"))
  1568  			})
  1569  		})
  1570  
  1571  		When("application exists", func() {
  1572  			var ccv3App resources.Application
  1573  
  1574  			BeforeEach(func() {
  1575  				ccv3App = resources.Application{
  1576  					GUID: "some-app-guid",
  1577  				}
  1578  
  1579  				fakeCloudControllerClient.GetApplicationsReturns(
  1580  					[]resources.Application{ccv3App},
  1581  					ccv3.Warnings{"some-warning"},
  1582  					nil,
  1583  				)
  1584  			})
  1585  
  1586  			When("setting the health check returns an error", func() {
  1587  				var expectedErr error
  1588  
  1589  				BeforeEach(func() {
  1590  					expectedErr = errors.New("some-error")
  1591  					fakeCloudControllerClient.GetApplicationProcessByTypeReturns(
  1592  						resources.Process{},
  1593  						ccv3.Warnings{"some-process-warning"},
  1594  						expectedErr,
  1595  					)
  1596  				})
  1597  
  1598  				It("returns the error and warnings", func() {
  1599  					Expect(err).To(Equal(expectedErr))
  1600  					Expect(warnings).To(ConsistOf("some-warning", "some-process-warning"))
  1601  				})
  1602  			})
  1603  
  1604  			When("application process exists", func() {
  1605  				BeforeEach(func() {
  1606  					fakeCloudControllerClient.GetApplicationProcessByTypeReturns(
  1607  						resources.Process{GUID: "some-process-guid"},
  1608  						ccv3.Warnings{"some-process-warning"},
  1609  						nil,
  1610  					)
  1611  
  1612  					fakeCloudControllerClient.UpdateProcessReturns(
  1613  						resources.Process{GUID: "some-process-guid"},
  1614  						ccv3.Warnings{"some-health-check-warning"},
  1615  						nil,
  1616  					)
  1617  				})
  1618  
  1619  				It("returns the application", func() {
  1620  					Expect(err).NotTo(HaveOccurred())
  1621  					Expect(warnings).To(ConsistOf("some-warning", "some-process-warning", "some-health-check-warning"))
  1622  
  1623  					Expect(app).To(Equal(resources.Application{
  1624  						GUID: ccv3App.GUID,
  1625  					}))
  1626  
  1627  					Expect(fakeCloudControllerClient.GetApplicationProcessByTypeCallCount()).To(Equal(1))
  1628  					appGUID, processType := fakeCloudControllerClient.GetApplicationProcessByTypeArgsForCall(0)
  1629  					Expect(appGUID).To(Equal("some-app-guid"))
  1630  					Expect(processType).To(Equal("some-process-type"))
  1631  
  1632  					Expect(fakeCloudControllerClient.UpdateProcessCallCount()).To(Equal(1))
  1633  					process := fakeCloudControllerClient.UpdateProcessArgsForCall(0)
  1634  					Expect(process.GUID).To(Equal("some-process-guid"))
  1635  					Expect(process.HealthCheckType).To(Equal(constant.HTTP))
  1636  					Expect(process.HealthCheckEndpoint).To(Equal("some-http-endpoint"))
  1637  					Expect(process.HealthCheckInvocationTimeout).To(BeEquivalentTo(42))
  1638  				})
  1639  			})
  1640  		})
  1641  	})
  1642  
  1643  	Describe("StopApplication", func() {
  1644  		var (
  1645  			warnings   Warnings
  1646  			executeErr error
  1647  		)
  1648  
  1649  		JustBeforeEach(func() {
  1650  			warnings, executeErr = actor.StopApplication("some-app-guid")
  1651  		})
  1652  
  1653  		When("there are no client errors", func() {
  1654  			BeforeEach(func() {
  1655  				fakeCloudControllerClient.UpdateApplicationStopReturns(
  1656  					resources.Application{GUID: "some-app-guid"},
  1657  					ccv3.Warnings{"stop-application-warning"},
  1658  					nil,
  1659  				)
  1660  			})
  1661  
  1662  			It("stops the application", func() {
  1663  				Expect(executeErr).ToNot(HaveOccurred())
  1664  				Expect(warnings).To(ConsistOf("stop-application-warning"))
  1665  
  1666  				Expect(fakeCloudControllerClient.UpdateApplicationStopCallCount()).To(Equal(1))
  1667  				Expect(fakeCloudControllerClient.UpdateApplicationStopArgsForCall(0)).To(Equal("some-app-guid"))
  1668  			})
  1669  		})
  1670  
  1671  		When("stopping the application fails", func() {
  1672  			var expectedErr error
  1673  			BeforeEach(func() {
  1674  				expectedErr = errors.New("some set stop-application error")
  1675  				fakeCloudControllerClient.UpdateApplicationStopReturns(
  1676  					resources.Application{},
  1677  					ccv3.Warnings{"stop-application-warning"},
  1678  					expectedErr,
  1679  				)
  1680  			})
  1681  
  1682  			It("returns the error", func() {
  1683  				Expect(executeErr).To(Equal(expectedErr))
  1684  				Expect(warnings).To(ConsistOf("stop-application-warning"))
  1685  			})
  1686  		})
  1687  	})
  1688  
  1689  	Describe("StartApplication", func() {
  1690  		var (
  1691  			warnings   Warnings
  1692  			executeErr error
  1693  		)
  1694  
  1695  		BeforeEach(func() {
  1696  			fakeConfig.StartupTimeoutReturns(time.Second)
  1697  			fakeConfig.PollingIntervalReturns(0)
  1698  		})
  1699  
  1700  		JustBeforeEach(func() {
  1701  			warnings, executeErr = actor.StartApplication("some-app-guid")
  1702  		})
  1703  
  1704  		When("there are no client errors", func() {
  1705  			BeforeEach(func() {
  1706  				fakeCloudControllerClient.UpdateApplicationStartReturns(
  1707  					resources.Application{GUID: "some-app-guid"},
  1708  					ccv3.Warnings{"start-application-warning"},
  1709  					nil,
  1710  				)
  1711  			})
  1712  
  1713  			It("starts the application", func() {
  1714  				Expect(executeErr).ToNot(HaveOccurred())
  1715  				Expect(warnings).To(ConsistOf("start-application-warning"))
  1716  
  1717  				Expect(fakeCloudControllerClient.UpdateApplicationStartCallCount()).To(Equal(1))
  1718  				Expect(fakeCloudControllerClient.UpdateApplicationStartArgsForCall(0)).To(Equal("some-app-guid"))
  1719  			})
  1720  		})
  1721  
  1722  		When("starting the application fails", func() {
  1723  			var expectedErr error
  1724  
  1725  			BeforeEach(func() {
  1726  				expectedErr = errors.New("some set start-application error")
  1727  				fakeCloudControllerClient.UpdateApplicationStartReturns(
  1728  					resources.Application{},
  1729  					ccv3.Warnings{"start-application-warning"},
  1730  					expectedErr,
  1731  				)
  1732  			})
  1733  
  1734  			It("returns the error", func() {
  1735  				warnings, err := actor.StartApplication("some-app-guid")
  1736  
  1737  				Expect(err).To(Equal(expectedErr))
  1738  				Expect(warnings).To(ConsistOf("start-application-warning"))
  1739  			})
  1740  		})
  1741  	})
  1742  
  1743  	Describe("RestartApplication", func() {
  1744  		var (
  1745  			warnings   Warnings
  1746  			executeErr error
  1747  			noWait     bool
  1748  		)
  1749  
  1750  		BeforeEach(func() {
  1751  			fakeConfig.StartupTimeoutReturns(time.Second)
  1752  			fakeConfig.PollingIntervalReturns(0)
  1753  			noWait = false
  1754  		})
  1755  
  1756  		JustBeforeEach(func() {
  1757  			warnings, executeErr = actor.RestartApplication("some-app-guid", noWait)
  1758  		})
  1759  
  1760  		When("restarting the application is successful", func() {
  1761  			BeforeEach(func() {
  1762  				fakeCloudControllerClient.UpdateApplicationRestartReturns(
  1763  					resources.Application{GUID: "some-app-guid"},
  1764  					ccv3.Warnings{"restart-application-warning"},
  1765  					nil,
  1766  				)
  1767  			})
  1768  
  1769  			It("does not error", func() {
  1770  				Expect(executeErr).ToNot(HaveOccurred())
  1771  				Expect(warnings).To(ConsistOf("restart-application-warning"))
  1772  			})
  1773  		})
  1774  
  1775  		When("restarting the application fails", func() {
  1776  			var expectedErr error
  1777  
  1778  			BeforeEach(func() {
  1779  				expectedErr = errors.New("some set restart-application error")
  1780  				fakeCloudControllerClient.UpdateApplicationRestartReturns(
  1781  					resources.Application{},
  1782  					ccv3.Warnings{"restart-application-warning"},
  1783  					expectedErr,
  1784  				)
  1785  			})
  1786  
  1787  			It("returns the warnings and error", func() {
  1788  				Expect(executeErr).To(Equal(expectedErr))
  1789  				Expect(warnings).To(ConsistOf("restart-application-warning"))
  1790  			})
  1791  		})
  1792  	})
  1793  
  1794  	Describe("PollProcesses", func() {
  1795  		var (
  1796  			processes               []resources.Process
  1797  			handleInstanceDetails   func(string)
  1798  			reportedInstanceDetails []string
  1799  
  1800  			keepPolling bool
  1801  			warnings    Warnings
  1802  			executeErr  error
  1803  		)
  1804  
  1805  		BeforeEach(func() {
  1806  			reportedInstanceDetails = []string{}
  1807  			handleInstanceDetails = func(instanceDetails string) {
  1808  				reportedInstanceDetails = append(reportedInstanceDetails, instanceDetails)
  1809  			}
  1810  
  1811  			processes = []resources.Process{
  1812  				{GUID: "process-1"},
  1813  				{GUID: "process-2"},
  1814  			}
  1815  		})
  1816  
  1817  		JustBeforeEach(func() {
  1818  			keepPolling, warnings, executeErr = actor.PollProcesses(processes, handleInstanceDetails)
  1819  		})
  1820  
  1821  		It("gets process instances for each process", func() {
  1822  			Expect(executeErr).NotTo(HaveOccurred())
  1823  			Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(2))
  1824  			Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process-1"))
  1825  			Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(1)).To(Equal("process-2"))
  1826  		})
  1827  
  1828  		When("getting the process instances fails", func() {
  1829  			BeforeEach(func() {
  1830  				fakeCloudControllerClient.GetProcessInstancesReturns(nil, ccv3.Warnings{"get-instances-warning"}, errors.New("get-instances-error"))
  1831  			})
  1832  
  1833  			It("returns an error and warnings and terminates the loop", func() {
  1834  				Expect(executeErr).To(MatchError("get-instances-error"))
  1835  				Expect(warnings).To(ConsistOf("get-instances-warning"))
  1836  				Expect(keepPolling).To(BeTrue())
  1837  
  1838  				Expect(fakeCloudControllerClient.GetProcessInstancesCallCount()).To(Equal(1))
  1839  				Expect(fakeCloudControllerClient.GetProcessInstancesArgsForCall(0)).To(Equal("process-1"))
  1840  			})
  1841  		})
  1842  
  1843  		When("getting the process instances is always successful", func() {
  1844  			When("a process has all instances crashed", func() {
  1845  				BeforeEach(func() {
  1846  					fakeCloudControllerClient.GetProcessInstancesReturns(
  1847  						[]ccv3.ProcessInstance{
  1848  							{State: constant.ProcessInstanceCrashed, Details: "details1"},
  1849  						},
  1850  						ccv3.Warnings{"get-process1-instances-warning"},
  1851  						nil,
  1852  					)
  1853  				})
  1854  
  1855  				It("calls the callback function with the retrieved instances", func() {
  1856  					Expect(reportedInstanceDetails).To(Equal([]string{
  1857  						"Error starting instances: 'details1'",
  1858  					}))
  1859  				})
  1860  
  1861  				It("returns an all instances crashed error", func() {
  1862  					Expect(executeErr).To(MatchError(actionerror.AllInstancesCrashedError{}))
  1863  					Expect(warnings).To(ConsistOf("get-process1-instances-warning"))
  1864  					Expect(keepPolling).To(BeTrue())
  1865  				})
  1866  			})
  1867  
  1868  			When("there are still instances in the starting state for a process", func() {
  1869  				BeforeEach(func() {
  1870  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(0,
  1871  						[]ccv3.ProcessInstance{
  1872  							{State: constant.ProcessInstanceRunning},
  1873  						},
  1874  						ccv3.Warnings{"get-process1-instances-warning"},
  1875  						nil,
  1876  					)
  1877  
  1878  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(1,
  1879  						[]ccv3.ProcessInstance{
  1880  							{State: constant.ProcessInstanceStarting, Details: "details2"},
  1881  						},
  1882  						ccv3.Warnings{"get-process2-instances-warning"},
  1883  						nil,
  1884  					)
  1885  				})
  1886  
  1887  				It("calls the callback function with the retrieved instances", func() {
  1888  					Expect(reportedInstanceDetails).To(Equal([]string{
  1889  						"Instances starting...",
  1890  						"Error starting instances: 'details2'",
  1891  					}))
  1892  				})
  1893  
  1894  				It("returns success and that we should keep polling", func() {
  1895  					Expect(executeErr).NotTo(HaveOccurred())
  1896  					Expect(warnings).To(ConsistOf("get-process1-instances-warning", "get-process2-instances-warning"))
  1897  					Expect(keepPolling).To(BeFalse())
  1898  				})
  1899  			})
  1900  
  1901  			When("all the instances of all processes are stable", func() {
  1902  				BeforeEach(func() {
  1903  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(0,
  1904  						[]ccv3.ProcessInstance{
  1905  							{State: constant.ProcessInstanceRunning, Details: "details1"},
  1906  						},
  1907  						ccv3.Warnings{"get-process1-instances-warning"},
  1908  						nil,
  1909  					)
  1910  
  1911  					fakeCloudControllerClient.GetProcessInstancesReturnsOnCall(1,
  1912  						[]ccv3.ProcessInstance{
  1913  							{State: constant.ProcessInstanceRunning},
  1914  						},
  1915  						ccv3.Warnings{"get-process2-instances-warning"},
  1916  						nil,
  1917  					)
  1918  				})
  1919  
  1920  				It("calls the callback function with the retrieved instances", func() {
  1921  					Expect(reportedInstanceDetails).To(Equal([]string{
  1922  						"Error starting instances: 'details1'",
  1923  						"Instances starting...",
  1924  					}))
  1925  				})
  1926  
  1927  				It("returns success and that we should keep polling", func() {
  1928  					Expect(executeErr).NotTo(HaveOccurred())
  1929  					Expect(warnings).To(ConsistOf("get-process1-instances-warning", "get-process2-instances-warning"))
  1930  					Expect(keepPolling).To(BeTrue())
  1931  				})
  1932  
  1933  			})
  1934  		})
  1935  
  1936  	})
  1937  
  1938  	Describe("GetUnstagedNewestPackageGUID", func() {
  1939  		var (
  1940  			packageToStage string
  1941  			warnings       Warnings
  1942  			executeErr     error
  1943  		)
  1944  
  1945  		JustBeforeEach(func() {
  1946  			packageToStage, warnings, executeErr = actor.GetUnstagedNewestPackageGUID("some-app-guid")
  1947  		})
  1948  
  1949  		// Nothing to stage.
  1950  		When("There are no packages on the app", func() {
  1951  			When("getting the packages succeeds", func() {
  1952  				BeforeEach(func() {
  1953  					fakeCloudControllerClient.GetPackagesReturns([]resources.Package{}, ccv3.Warnings{"get-packages-warnings"}, nil)
  1954  				})
  1955  
  1956  				It("checks for packages", func() {
  1957  					Expect(fakeCloudControllerClient.GetPackagesCallCount()).To(Equal(1))
  1958  					Expect(fakeCloudControllerClient.GetPackagesArgsForCall(0)).To(ConsistOf(
  1959  						ccv3.Query{Key: ccv3.AppGUIDFilter, Values: []string{"some-app-guid"}},
  1960  						ccv3.Query{Key: ccv3.OrderBy, Values: []string{ccv3.CreatedAtDescendingOrder}},
  1961  						ccv3.Query{Key: ccv3.PerPage, Values: []string{"1"}},
  1962  					))
  1963  				})
  1964  
  1965  				It("returns empty string", func() {
  1966  					Expect(packageToStage).To(Equal(""))
  1967  					Expect(warnings).To(ConsistOf("get-packages-warnings"))
  1968  					Expect(executeErr).To(BeNil())
  1969  				})
  1970  			})
  1971  
  1972  			When("getting the packages fails", func() {
  1973  				BeforeEach(func() {
  1974  					fakeCloudControllerClient.GetPackagesReturns(
  1975  						nil,
  1976  						ccv3.Warnings{"get-packages-warnings"},
  1977  						errors.New("get-packages-error"),
  1978  					)
  1979  				})
  1980  
  1981  				It("returns the error", func() {
  1982  					Expect(warnings).To(ConsistOf("get-packages-warnings"))
  1983  					Expect(executeErr).To(MatchError("get-packages-error"))
  1984  				})
  1985  			})
  1986  		})
  1987  
  1988  		When("there are packages", func() {
  1989  			BeforeEach(func() {
  1990  				fakeCloudControllerClient.GetPackagesReturns(
  1991  					[]resources.Package{{GUID: "package-guid", CreatedAt: "2019-01-01T06:00:00Z"}},
  1992  					ccv3.Warnings{"get-packages-warning"},
  1993  					nil)
  1994  			})
  1995  
  1996  			It("checks for the packages latest droplet", func() {
  1997  				Expect(fakeCloudControllerClient.GetPackageDropletsCallCount()).To(Equal(1))
  1998  				packageGuid, queries := fakeCloudControllerClient.GetPackageDropletsArgsForCall(0)
  1999  				Expect(packageGuid).To(Equal("package-guid"))
  2000  				Expect(queries).To(ConsistOf(
  2001  					ccv3.Query{Key: ccv3.PerPage, Values: []string{"1"}},
  2002  					ccv3.Query{Key: ccv3.StatesFilter, Values: []string{"STAGED"}},
  2003  				))
  2004  			})
  2005  
  2006  			When("the newest package's has a STAGED droplet", func() {
  2007  				BeforeEach(func() {
  2008  					fakeCloudControllerClient.GetPackageDropletsReturns(
  2009  						[]resources.Droplet{{State: constant.DropletStaged}},
  2010  						ccv3.Warnings{"get-package-droplet-warning"},
  2011  						nil,
  2012  					)
  2013  				})
  2014  
  2015  				It("returns empty string", func() {
  2016  					Expect(packageToStage).To(Equal(""))
  2017  					Expect(warnings).To(ConsistOf("get-packages-warning", "get-package-droplet-warning"))
  2018  					Expect(executeErr).To(BeNil())
  2019  				})
  2020  			})
  2021  
  2022  			When("the package has no STAGED droplets", func() {
  2023  				BeforeEach(func() {
  2024  					fakeCloudControllerClient.GetPackageDropletsReturns(
  2025  						[]resources.Droplet{},
  2026  						ccv3.Warnings{"get-package-droplet-warning"},
  2027  						nil,
  2028  					)
  2029  				})
  2030  
  2031  				It("returns the guid of the newest package", func() {
  2032  					Expect(packageToStage).To(Equal("package-guid"))
  2033  					Expect(warnings).To(ConsistOf("get-packages-warning", "get-package-droplet-warning"))
  2034  					Expect(executeErr).To(BeNil())
  2035  				})
  2036  			})
  2037  		})
  2038  	})
  2039  
  2040  	Describe("RenameApplicationByNameAndSpaceGUID", func() {
  2041  		When("the app does not exist", func() {
  2042  			BeforeEach(func() {
  2043  				fakeCloudControllerClient.GetApplicationsReturns(
  2044  					[]resources.Application{},
  2045  					ccv3.Warnings{"some-warning"},
  2046  					nil,
  2047  				)
  2048  			})
  2049  
  2050  			It("returns an ApplicationNotFoundError and the warnings", func() {
  2051  				_, warnings, err := actor.RenameApplicationByNameAndSpaceGUID("old-app-name", "new-app-name", "space-guid")
  2052  				Expect(warnings).To(ConsistOf("some-warning"))
  2053  				Expect(err).To(MatchError(actionerror.ApplicationNotFoundError{Name: "old-app-name"}))
  2054  			})
  2055  		})
  2056  
  2057  		When("the cloud controller client returns an error on application find", func() {
  2058  			var expectedError error
  2059  
  2060  			BeforeEach(func() {
  2061  				expectedError = errors.New("I am a CloudControllerClient Error")
  2062  				fakeCloudControllerClient.GetApplicationsReturns(
  2063  					[]resources.Application{},
  2064  					ccv3.Warnings{"some-warning"},
  2065  					expectedError)
  2066  			})
  2067  
  2068  			It("returns the warnings and the error", func() {
  2069  				_, warnings, err := actor.RenameApplicationByNameAndSpaceGUID("old-app-name", "new-app-name", "space-guid")
  2070  				Expect(warnings).To(ConsistOf("some-warning"))
  2071  				Expect(err).To(MatchError(expectedError))
  2072  			})
  2073  		})
  2074  
  2075  		When("the cloud controller client returns an error on application update", func() {
  2076  			var expectedError error
  2077  
  2078  			BeforeEach(func() {
  2079  				expectedError = errors.New("I am a CloudControllerClient Error")
  2080  				fakeCloudControllerClient.GetApplicationsReturns(
  2081  					[]resources.Application{
  2082  						{
  2083  							Name: "old-app-name",
  2084  							GUID: "old-app-guid",
  2085  						},
  2086  					},
  2087  					ccv3.Warnings{"get-app-warning"},
  2088  					nil)
  2089  				fakeCloudControllerClient.UpdateApplicationNameReturns(
  2090  					resources.Application{},
  2091  					ccv3.Warnings{"update-app-warning"},
  2092  					expectedError)
  2093  			})
  2094  
  2095  			It("returns the warnings and the error", func() {
  2096  				_, warnings, err := actor.RenameApplicationByNameAndSpaceGUID("old-app-name", "new-app-name", "space-guid")
  2097  				Expect(warnings).To(ConsistOf("get-app-warning", "update-app-warning"))
  2098  				Expect(err).To(MatchError(expectedError))
  2099  			})
  2100  		})
  2101  
  2102  		When("the app exists", func() {
  2103  			BeforeEach(func() {
  2104  				fakeCloudControllerClient.GetApplicationsReturns(
  2105  					[]resources.Application{
  2106  						{
  2107  							Name: "old-app-name",
  2108  							GUID: "old-app-guid",
  2109  						},
  2110  					},
  2111  					ccv3.Warnings{"get-app-warning"},
  2112  					nil,
  2113  				)
  2114  
  2115  				fakeCloudControllerClient.UpdateApplicationNameReturns(
  2116  					resources.Application{
  2117  						Name: "new-app-name",
  2118  						GUID: "old-app-guid",
  2119  					},
  2120  					ccv3.Warnings{"update-app-warning"},
  2121  					nil,
  2122  				)
  2123  			})
  2124  
  2125  			It("changes the app name and returns the application and warnings", func() {
  2126  				app, warnings, err := actor.RenameApplicationByNameAndSpaceGUID("old-app-name", "new-app-name", "some-space-guid")
  2127  				Expect(err).ToNot(HaveOccurred())
  2128  				Expect(app).To(Equal(resources.Application{
  2129  					Name: "new-app-name",
  2130  					GUID: "old-app-guid",
  2131  				}))
  2132  				Expect(warnings).To(ConsistOf("get-app-warning", "update-app-warning"))
  2133  				appName, appGuid := fakeCloudControllerClient.UpdateApplicationNameArgsForCall(0)
  2134  				Expect(appName).To(Equal("new-app-name"))
  2135  				Expect(appGuid).To(Equal("old-app-guid"))
  2136  			})
  2137  		})
  2138  
  2139  	})
  2140  })