All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.netflix.spinnaker.clouddriver.google.provider.agent.GoogleImageCachingAgent.groovy Maven / Gradle / Ivy

/*
 * Copyright 2016 Google, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License")
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.netflix.spinnaker.clouddriver.google.provider.agent

import com.fasterxml.jackson.databind.ObjectMapper
import com.google.api.client.googleapis.batch.json.JsonBatchCallback
import com.google.api.client.googleapis.json.GoogleJsonError
import com.google.api.client.http.HttpHeaders
import com.google.api.services.compute.Compute
import com.google.api.services.compute.model.Image
import com.google.api.services.compute.model.ImageList
import com.google.common.annotations.VisibleForTesting
import com.netflix.spectator.api.Registry
import com.netflix.spinnaker.cats.agent.AgentDataType
import com.netflix.spinnaker.cats.agent.CacheResult
import com.netflix.spinnaker.cats.provider.ProviderCache
import com.netflix.spinnaker.clouddriver.google.cache.CacheResultBuilder
import com.netflix.spinnaker.clouddriver.google.cache.Keys
import com.netflix.spinnaker.clouddriver.google.deploy.GCEUtil
import com.netflix.spinnaker.clouddriver.google.batch.GoogleBatchRequest
import com.netflix.spinnaker.clouddriver.google.security.GoogleNamedAccountCredentials
import groovy.util.logging.Slf4j
import org.slf4j.LoggerFactory

import static com.netflix.spinnaker.cats.agent.AgentDataType.Authority.AUTHORITATIVE
import static com.netflix.spinnaker.clouddriver.google.cache.Keys.Namespace.IMAGES

@Slf4j
class GoogleImageCachingAgent extends AbstractGoogleCachingAgent {

  final Set providedDataTypes = [
      AUTHORITATIVE.forType(IMAGES.ns)
  ] as Set

  String agentType = "$accountName/$GoogleImageCachingAgent.simpleName"

  List imageProjects
  List baseImageProjects

  @VisibleForTesting
  GoogleImageCachingAgent() {}

  GoogleImageCachingAgent(String clouddriverUserAgentApplicationName,
                          GoogleNamedAccountCredentials credentials,
                          ObjectMapper objectMapper,
                          Registry registry,
                          List imageProjects,
                          List baseImageProjects) {
    super(clouddriverUserAgentApplicationName,
          credentials,
          objectMapper,
          registry)
    this.imageProjects = imageProjects
    this.baseImageProjects = baseImageProjects
  }

  @Override
  CacheResult loadData(ProviderCache providerCache) {
    List imageList = loadImages()
    buildCacheResult(providerCache, imageList)
  }

  List loadImages() {
    List imageList = []
    List allImageProjects = [project] + imageProjects + baseImageProjects - null

    // We want predictable iteration order that matches the order of insertion.
    LinkedHashMap imageProjectToNextPageTokenMap = new LinkedHashMap<>()

    // This will ensure that each image project is queried.
    allImageProjects.each { imageProjectToNextPageTokenMap[it] = null }

    while (imageProjectToNextPageTokenMap) {
      GoogleBatchRequest imageListBatch = buildGoogleBatchRequest()
      AllImagesCallback imageListCallback =
        new AllImagesCallback(imageProjectToNextPageTokenMap: imageProjectToNextPageTokenMap, imageList: imageList)

      imageProjectToNextPageTokenMap.each { imageProject, pageToken ->
        Compute.Images.List imagesList = compute.images().list(imageProject)

        if (pageToken) {
          imagesList = imagesList.setPageToken(pageToken)
        }

        imageListBatch.queue(imagesList, imageListCallback)
      }

      executeIfRequestsAreQueued(imageListBatch, "ImageCaching.image")
    }

    return imageList
  }

  private CacheResult buildCacheResult(ProviderCache _, List imageList) {
    log.debug("Describing items in ${agentType}")

    def cacheResultBuilder = new CacheResultBuilder()

    imageList.each { Image image ->
      def imageKey = Keys.getImageKey(accountName, image.getName())

      cacheResultBuilder.namespace(IMAGES.ns).keep(imageKey).with {
        attributes.image = image
      }
    }

    log.debug("Caching ${cacheResultBuilder.namespace(IMAGES.ns).keepSize()} items in ${agentType}")

    cacheResultBuilder.build()
  }

  static List filterDeprecatedImages(ImageList imageListResult) {
    imageListResult?.items?.findAll { Image image ->
      !image.deprecated?.state
    } ?: []
  }

  class AllImagesCallback extends JsonBatchCallback implements FailureLogger {

    LinkedHashMap imageProjectToNextPageTokenMap
    List imageList

    @Override
    void onSuccess(ImageList imageListResult, HttpHeaders responseHeaders) throws IOException {
      def nonDeprecatedImages = filterDeprecatedImages(imageListResult)
      if (nonDeprecatedImages) {
        imageList.addAll(nonDeprecatedImages)
      }

      def imageProject = GCEUtil.deriveProjectId(imageListResult.getSelfLink())

      if (imageListResult.nextPageToken) {
        imageProjectToNextPageTokenMap[imageProject] = imageListResult.nextPageToken
      } else {
        imageProjectToNextPageTokenMap.remove(imageProject)
      }
    }

    @Override
    void onFailure(GoogleJsonError e, HttpHeaders responseHeaders) throws IOException {
      LoggerFactory.getLogger(this.class).error e.getMessage()
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy