github.com/codemac/docker@v1.2.1-0.20150518222241-6a18412d5b9c/graph/pull.go (about) 1 package graph 2 3 import ( 4 "fmt" 5 "io" 6 "io/ioutil" 7 "net" 8 "net/url" 9 "os" 10 "strings" 11 "time" 12 13 "github.com/Sirupsen/logrus" 14 "github.com/docker/distribution/digest" 15 "github.com/docker/docker/cliconfig" 16 "github.com/docker/docker/image" 17 "github.com/docker/docker/pkg/progressreader" 18 "github.com/docker/docker/pkg/streamformatter" 19 "github.com/docker/docker/pkg/stringid" 20 "github.com/docker/docker/registry" 21 "github.com/docker/docker/utils" 22 ) 23 24 type ImagePullConfig struct { 25 MetaHeaders map[string][]string 26 AuthConfig *cliconfig.AuthConfig 27 OutStream io.Writer 28 } 29 30 func (s *TagStore) Pull(image string, tag string, imagePullConfig *ImagePullConfig) error { 31 var ( 32 sf = streamformatter.NewJSONStreamFormatter() 33 ) 34 35 // Resolve the Repository name from fqn to RepositoryInfo 36 repoInfo, err := s.registryService.ResolveRepository(image) 37 if err != nil { 38 return err 39 } 40 41 if err := validateRepoName(repoInfo.LocalName); err != nil { 42 return err 43 } 44 45 c, err := s.poolAdd("pull", utils.ImageReference(repoInfo.LocalName, tag)) 46 if err != nil { 47 if c != nil { 48 // Another pull of the same repository is already taking place; just wait for it to finish 49 imagePullConfig.OutStream.Write(sf.FormatStatus("", "Repository %s already being pulled by another client. Waiting.", repoInfo.LocalName)) 50 <-c 51 return nil 52 } 53 return err 54 } 55 defer s.poolRemove("pull", utils.ImageReference(repoInfo.LocalName, tag)) 56 57 logrus.Debugf("pulling image from host %q with remote name %q", repoInfo.Index.Name, repoInfo.RemoteName) 58 endpoint, err := repoInfo.GetEndpoint() 59 if err != nil { 60 return err 61 } 62 63 r, err := registry.NewSession(imagePullConfig.AuthConfig, registry.HTTPRequestFactory(imagePullConfig.MetaHeaders), endpoint, true) 64 if err != nil { 65 return err 66 } 67 68 logName := repoInfo.LocalName 69 if tag != "" { 70 logName = utils.ImageReference(logName, tag) 71 } 72 73 if len(repoInfo.Index.Mirrors) == 0 && (repoInfo.Index.Official || endpoint.Version == registry.APIVersion2) { 74 if repoInfo.Official { 75 s.trustService.UpdateBase() 76 } 77 78 logrus.Debugf("pulling v2 repository with local name %q", repoInfo.LocalName) 79 if err := s.pullV2Repository(r, imagePullConfig.OutStream, repoInfo, tag, sf); err == nil { 80 s.eventsService.Log("pull", logName, "") 81 return nil 82 } else if err != registry.ErrDoesNotExist && err != ErrV2RegistryUnavailable { 83 logrus.Errorf("Error from V2 registry: %s", err) 84 } 85 86 logrus.Debug("image does not exist on v2 registry, falling back to v1") 87 } 88 89 logrus.Debugf("pulling v1 repository with local name %q", repoInfo.LocalName) 90 if err = s.pullRepository(r, imagePullConfig.OutStream, repoInfo, tag, sf); err != nil { 91 return err 92 } 93 94 s.eventsService.Log("pull", logName, "") 95 96 return nil 97 } 98 99 func (s *TagStore) pullRepository(r *registry.Session, out io.Writer, repoInfo *registry.RepositoryInfo, askedTag string, sf *streamformatter.StreamFormatter) error { 100 out.Write(sf.FormatStatus("", "Pulling repository %s", repoInfo.CanonicalName)) 101 102 repoData, err := r.GetRepositoryData(repoInfo.RemoteName) 103 if err != nil { 104 if strings.Contains(err.Error(), "HTTP code: 404") { 105 return fmt.Errorf("Error: image %s not found", utils.ImageReference(repoInfo.RemoteName, askedTag)) 106 } 107 // Unexpected HTTP error 108 return err 109 } 110 111 logrus.Debugf("Retrieving the tag list") 112 tagsList, err := r.GetRemoteTags(repoData.Endpoints, repoInfo.RemoteName, repoData.Tokens) 113 if err != nil { 114 logrus.Errorf("unable to get remote tags: %s", err) 115 return err 116 } 117 118 for tag, id := range tagsList { 119 repoData.ImgList[id] = ®istry.ImgData{ 120 ID: id, 121 Tag: tag, 122 Checksum: "", 123 } 124 } 125 126 logrus.Debugf("Registering tags") 127 // If no tag has been specified, pull them all 128 if askedTag == "" { 129 for tag, id := range tagsList { 130 repoData.ImgList[id].Tag = tag 131 } 132 } else { 133 // Otherwise, check that the tag exists and use only that one 134 id, exists := tagsList[askedTag] 135 if !exists { 136 return fmt.Errorf("Tag %s not found in repository %s", askedTag, repoInfo.CanonicalName) 137 } 138 repoData.ImgList[id].Tag = askedTag 139 } 140 141 errors := make(chan error) 142 143 layersDownloaded := false 144 for _, image := range repoData.ImgList { 145 downloadImage := func(img *registry.ImgData) { 146 if askedTag != "" && img.Tag != askedTag { 147 errors <- nil 148 return 149 } 150 151 if img.Tag == "" { 152 logrus.Debugf("Image (id: %s) present in this repository but untagged, skipping", img.ID) 153 errors <- nil 154 return 155 } 156 157 // ensure no two downloads of the same image happen at the same time 158 if c, err := s.poolAdd("pull", "img:"+img.ID); err != nil { 159 if c != nil { 160 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Layer already being pulled by another client. Waiting.", nil)) 161 <-c 162 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Download complete", nil)) 163 } else { 164 logrus.Debugf("Image (id: %s) pull is already running, skipping: %v", img.ID, err) 165 } 166 errors <- nil 167 return 168 } 169 defer s.poolRemove("pull", "img:"+img.ID) 170 171 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), fmt.Sprintf("Pulling image (%s) from %s", img.Tag, repoInfo.CanonicalName), nil)) 172 success := false 173 var lastErr, err error 174 var isDownloaded bool 175 for _, ep := range repoInfo.Index.Mirrors { 176 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), fmt.Sprintf("Pulling image (%s) from %s, mirror: %s", img.Tag, repoInfo.CanonicalName, ep), nil)) 177 if isDownloaded, err = s.pullImage(r, out, img.ID, ep, repoData.Tokens, sf); err != nil { 178 // Don't report errors when pulling from mirrors. 179 logrus.Debugf("Error pulling image (%s) from %s, mirror: %s, %s", img.Tag, repoInfo.CanonicalName, ep, err) 180 continue 181 } 182 layersDownloaded = layersDownloaded || isDownloaded 183 success = true 184 break 185 } 186 if !success { 187 for _, ep := range repoData.Endpoints { 188 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), fmt.Sprintf("Pulling image (%s) from %s, endpoint: %s", img.Tag, repoInfo.CanonicalName, ep), nil)) 189 if isDownloaded, err = s.pullImage(r, out, img.ID, ep, repoData.Tokens, sf); err != nil { 190 // It's not ideal that only the last error is returned, it would be better to concatenate the errors. 191 // As the error is also given to the output stream the user will see the error. 192 lastErr = err 193 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), fmt.Sprintf("Error pulling image (%s) from %s, endpoint: %s, %s", img.Tag, repoInfo.CanonicalName, ep, err), nil)) 194 continue 195 } 196 layersDownloaded = layersDownloaded || isDownloaded 197 success = true 198 break 199 } 200 } 201 if !success { 202 err := fmt.Errorf("Error pulling image (%s) from %s, %v", img.Tag, repoInfo.CanonicalName, lastErr) 203 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), err.Error(), nil)) 204 errors <- err 205 return 206 } 207 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Download complete", nil)) 208 209 errors <- nil 210 } 211 212 go downloadImage(image) 213 } 214 215 var lastError error 216 for i := 0; i < len(repoData.ImgList); i++ { 217 if err := <-errors; err != nil { 218 lastError = err 219 } 220 } 221 if lastError != nil { 222 return lastError 223 } 224 225 for tag, id := range tagsList { 226 if askedTag != "" && tag != askedTag { 227 continue 228 } 229 if err := s.Tag(repoInfo.LocalName, tag, id, true); err != nil { 230 return err 231 } 232 } 233 234 requestedTag := repoInfo.CanonicalName 235 if len(askedTag) > 0 { 236 requestedTag = utils.ImageReference(repoInfo.CanonicalName, askedTag) 237 } 238 WriteStatus(requestedTag, out, sf, layersDownloaded) 239 return nil 240 } 241 242 func (s *TagStore) pullImage(r *registry.Session, out io.Writer, imgID, endpoint string, token []string, sf *streamformatter.StreamFormatter) (bool, error) { 243 history, err := r.GetRemoteHistory(imgID, endpoint, token) 244 if err != nil { 245 return false, err 246 } 247 out.Write(sf.FormatProgress(stringid.TruncateID(imgID), "Pulling dependent layers", nil)) 248 // FIXME: Try to stream the images? 249 // FIXME: Launch the getRemoteImage() in goroutines 250 251 layersDownloaded := false 252 for i := len(history) - 1; i >= 0; i-- { 253 id := history[i] 254 255 // ensure no two downloads of the same layer happen at the same time 256 if c, err := s.poolAdd("pull", "layer:"+id); err != nil { 257 logrus.Debugf("Image (id: %s) pull is already running, skipping: %v", id, err) 258 <-c 259 } 260 defer s.poolRemove("pull", "layer:"+id) 261 262 if !s.graph.Exists(id) { 263 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Pulling metadata", nil)) 264 var ( 265 imgJSON []byte 266 imgSize int 267 err error 268 img *image.Image 269 ) 270 retries := 5 271 for j := 1; j <= retries; j++ { 272 imgJSON, imgSize, err = r.GetRemoteImageJSON(id, endpoint, token) 273 if err != nil && j == retries { 274 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Error pulling dependent layers", nil)) 275 return layersDownloaded, err 276 } else if err != nil { 277 time.Sleep(time.Duration(j) * 500 * time.Millisecond) 278 continue 279 } 280 img, err = image.NewImgJSON(imgJSON) 281 layersDownloaded = true 282 if err != nil && j == retries { 283 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Error pulling dependent layers", nil)) 284 return layersDownloaded, fmt.Errorf("Failed to parse json: %s", err) 285 } else if err != nil { 286 time.Sleep(time.Duration(j) * 500 * time.Millisecond) 287 continue 288 } else { 289 break 290 } 291 } 292 293 for j := 1; j <= retries; j++ { 294 // Get the layer 295 status := "Pulling fs layer" 296 if j > 1 { 297 status = fmt.Sprintf("Pulling fs layer [retries: %d]", j) 298 } 299 out.Write(sf.FormatProgress(stringid.TruncateID(id), status, nil)) 300 layer, err := r.GetRemoteImageLayer(img.ID, endpoint, token, int64(imgSize)) 301 if uerr, ok := err.(*url.Error); ok { 302 err = uerr.Err 303 } 304 if terr, ok := err.(net.Error); ok && terr.Timeout() && j < retries { 305 time.Sleep(time.Duration(j) * 500 * time.Millisecond) 306 continue 307 } else if err != nil { 308 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Error pulling dependent layers", nil)) 309 return layersDownloaded, err 310 } 311 layersDownloaded = true 312 defer layer.Close() 313 314 err = s.graph.Register(img, 315 progressreader.New(progressreader.Config{ 316 In: layer, 317 Out: out, 318 Formatter: sf, 319 Size: imgSize, 320 NewLines: false, 321 ID: stringid.TruncateID(id), 322 Action: "Downloading", 323 })) 324 if terr, ok := err.(net.Error); ok && terr.Timeout() && j < retries { 325 time.Sleep(time.Duration(j) * 500 * time.Millisecond) 326 continue 327 } else if err != nil { 328 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Error downloading dependent layers", nil)) 329 return layersDownloaded, err 330 } else { 331 break 332 } 333 } 334 } 335 out.Write(sf.FormatProgress(stringid.TruncateID(id), "Download complete", nil)) 336 } 337 return layersDownloaded, nil 338 } 339 340 func WriteStatus(requestedTag string, out io.Writer, sf *streamformatter.StreamFormatter, layersDownloaded bool) { 341 if layersDownloaded { 342 out.Write(sf.FormatStatus("", "Status: Downloaded newer image for %s", requestedTag)) 343 } else { 344 out.Write(sf.FormatStatus("", "Status: Image is up to date for %s", requestedTag)) 345 } 346 } 347 348 // downloadInfo is used to pass information from download to extractor 349 type downloadInfo struct { 350 imgJSON []byte 351 img *image.Image 352 digest digest.Digest 353 tmpFile *os.File 354 length int64 355 downloaded bool 356 err chan error 357 } 358 359 func (s *TagStore) pullV2Repository(r *registry.Session, out io.Writer, repoInfo *registry.RepositoryInfo, tag string, sf *streamformatter.StreamFormatter) error { 360 endpoint, err := r.V2RegistryEndpoint(repoInfo.Index) 361 if err != nil { 362 if repoInfo.Index.Official { 363 logrus.Debugf("Unable to pull from V2 registry, falling back to v1: %s", err) 364 return ErrV2RegistryUnavailable 365 } 366 return fmt.Errorf("error getting registry endpoint: %s", err) 367 } 368 auth, err := r.GetV2Authorization(endpoint, repoInfo.RemoteName, true) 369 if err != nil { 370 return fmt.Errorf("error getting authorization: %s", err) 371 } 372 var layersDownloaded bool 373 if tag == "" { 374 logrus.Debugf("Pulling tag list from V2 registry for %s", repoInfo.CanonicalName) 375 tags, err := r.GetV2RemoteTags(endpoint, repoInfo.RemoteName, auth) 376 if err != nil { 377 return err 378 } 379 if len(tags) == 0 { 380 return registry.ErrDoesNotExist 381 } 382 for _, t := range tags { 383 if downloaded, err := s.pullV2Tag(r, out, endpoint, repoInfo, t, sf, auth); err != nil { 384 return err 385 } else if downloaded { 386 layersDownloaded = true 387 } 388 } 389 } else { 390 if downloaded, err := s.pullV2Tag(r, out, endpoint, repoInfo, tag, sf, auth); err != nil { 391 return err 392 } else if downloaded { 393 layersDownloaded = true 394 } 395 } 396 397 requestedTag := repoInfo.CanonicalName 398 if len(tag) > 0 { 399 requestedTag = utils.ImageReference(repoInfo.CanonicalName, tag) 400 } 401 WriteStatus(requestedTag, out, sf, layersDownloaded) 402 return nil 403 } 404 405 func (s *TagStore) pullV2Tag(r *registry.Session, out io.Writer, endpoint *registry.Endpoint, repoInfo *registry.RepositoryInfo, tag string, sf *streamformatter.StreamFormatter, auth *registry.RequestAuthorization) (bool, error) { 406 logrus.Debugf("Pulling tag from V2 registry: %q", tag) 407 408 manifestBytes, manifestDigest, err := r.GetV2ImageManifest(endpoint, repoInfo.RemoteName, tag, auth) 409 if err != nil { 410 return false, err 411 } 412 413 // loadManifest ensures that the manifest payload has the expected digest 414 // if the tag is a digest reference. 415 manifest, verified, err := s.loadManifest(manifestBytes, manifestDigest, tag) 416 if err != nil { 417 return false, fmt.Errorf("error verifying manifest: %s", err) 418 } 419 420 if err := checkValidManifest(manifest); err != nil { 421 return false, err 422 } 423 424 if verified { 425 logrus.Printf("Image manifest for %s has been verified", utils.ImageReference(repoInfo.CanonicalName, tag)) 426 } 427 out.Write(sf.FormatStatus(tag, "Pulling from %s", repoInfo.CanonicalName)) 428 429 downloads := make([]downloadInfo, len(manifest.FSLayers)) 430 431 for i := len(manifest.FSLayers) - 1; i >= 0; i-- { 432 var ( 433 sumStr = manifest.FSLayers[i].BlobSum 434 imgJSON = []byte(manifest.History[i].V1Compatibility) 435 ) 436 437 img, err := image.NewImgJSON(imgJSON) 438 if err != nil { 439 return false, fmt.Errorf("failed to parse json: %s", err) 440 } 441 downloads[i].img = img 442 443 // Check if exists 444 if s.graph.Exists(img.ID) { 445 logrus.Debugf("Image already exists: %s", img.ID) 446 continue 447 } 448 449 dgst, err := digest.ParseDigest(sumStr) 450 if err != nil { 451 return false, err 452 } 453 downloads[i].digest = dgst 454 455 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Pulling fs layer", nil)) 456 457 downloadFunc := func(di *downloadInfo) error { 458 logrus.Debugf("pulling blob %q to V1 img %s", sumStr, img.ID) 459 460 if c, err := s.poolAdd("pull", "img:"+img.ID); err != nil { 461 if c != nil { 462 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Layer already being pulled by another client. Waiting.", nil)) 463 <-c 464 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Download complete", nil)) 465 } else { 466 logrus.Debugf("Image (id: %s) pull is already running, skipping: %v", img.ID, err) 467 } 468 } else { 469 defer s.poolRemove("pull", "img:"+img.ID) 470 tmpFile, err := ioutil.TempFile("", "GetV2ImageBlob") 471 if err != nil { 472 return err 473 } 474 475 r, l, err := r.GetV2ImageBlobReader(endpoint, repoInfo.RemoteName, di.digest, auth) 476 if err != nil { 477 return err 478 } 479 defer r.Close() 480 481 verifier, err := digest.NewDigestVerifier(di.digest) 482 if err != nil { 483 return err 484 } 485 486 if _, err := io.Copy(tmpFile, progressreader.New(progressreader.Config{ 487 In: ioutil.NopCloser(io.TeeReader(r, verifier)), 488 Out: out, 489 Formatter: sf, 490 Size: int(l), 491 NewLines: false, 492 ID: stringid.TruncateID(img.ID), 493 Action: "Downloading", 494 })); err != nil { 495 return fmt.Errorf("unable to copy v2 image blob data: %s", err) 496 } 497 498 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Verifying Checksum", nil)) 499 500 if !verifier.Verified() { 501 logrus.Infof("Image verification failed: checksum mismatch for %q", di.digest.String()) 502 verified = false 503 } 504 505 out.Write(sf.FormatProgress(stringid.TruncateID(img.ID), "Download complete", nil)) 506 507 logrus.Debugf("Downloaded %s to tempfile %s", img.ID, tmpFile.Name()) 508 di.tmpFile = tmpFile 509 di.length = l 510 di.downloaded = true 511 } 512 di.imgJSON = imgJSON 513 514 return nil 515 } 516 517 downloads[i].err = make(chan error) 518 go func(di *downloadInfo) { 519 di.err <- downloadFunc(di) 520 }(&downloads[i]) 521 } 522 523 var tagUpdated bool 524 for i := len(downloads) - 1; i >= 0; i-- { 525 d := &downloads[i] 526 if d.err != nil { 527 if err := <-d.err; err != nil { 528 return false, err 529 } 530 } 531 if d.downloaded { 532 // if tmpFile is empty assume download and extracted elsewhere 533 defer os.Remove(d.tmpFile.Name()) 534 defer d.tmpFile.Close() 535 d.tmpFile.Seek(0, 0) 536 if d.tmpFile != nil { 537 err = s.graph.Register(d.img, 538 progressreader.New(progressreader.Config{ 539 In: d.tmpFile, 540 Out: out, 541 Formatter: sf, 542 Size: int(d.length), 543 ID: stringid.TruncateID(d.img.ID), 544 Action: "Extracting", 545 })) 546 if err != nil { 547 return false, err 548 } 549 550 // FIXME: Pool release here for parallel tag pull (ensures any downloads block until fully extracted) 551 } 552 out.Write(sf.FormatProgress(stringid.TruncateID(d.img.ID), "Pull complete", nil)) 553 tagUpdated = true 554 } else { 555 out.Write(sf.FormatProgress(stringid.TruncateID(d.img.ID), "Already exists", nil)) 556 } 557 558 } 559 560 // Check for new tag if no layers downloaded 561 if !tagUpdated { 562 repo, err := s.Get(repoInfo.LocalName) 563 if err != nil { 564 return false, err 565 } 566 if repo != nil { 567 if _, exists := repo[tag]; !exists { 568 tagUpdated = true 569 } 570 } else { 571 tagUpdated = true 572 } 573 } 574 575 if verified && tagUpdated { 576 out.Write(sf.FormatStatus(utils.ImageReference(repoInfo.CanonicalName, tag), "The image you are pulling has been verified. Important: image verification is a tech preview feature and should not be relied on to provide security.")) 577 } 578 579 if manifestDigest != "" { 580 out.Write(sf.FormatStatus("", "Digest: %s", manifestDigest)) 581 } 582 583 if utils.DigestReference(tag) { 584 if err = s.SetDigest(repoInfo.LocalName, tag, downloads[0].img.ID); err != nil { 585 return false, err 586 } 587 } else { 588 // only set the repository/tag -> image ID mapping when pulling by tag (i.e. not by digest) 589 if err = s.Tag(repoInfo.LocalName, tag, downloads[0].img.ID, true); err != nil { 590 return false, err 591 } 592 } 593 594 return tagUpdated, nil 595 }