Rietveld Code Review Tool
Help | Bug tracker | Discussion group | Source code | Sign in
(15)

Side by Side Diff: state/apiserver/charms.go

Issue 49960047: charm: add Bundle.Manifest
Patch Set: charm: add Bundle.Manifest Created 11 years, 1 month ago
Left:
Right:
Use n/p to move between diff chunks; N/P to move between comments. Please Sign in to add in-line comments.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2013 Canonical Ltd. 1 // Copyright 2013 Canonical Ltd.
2 // Licensed under the AGPLv3, see LICENCE file for details. 2 // Licensed under the AGPLv3, see LICENCE file for details.
3 3
4 package apiserver 4 package apiserver
5 5
6 import ( 6 import (
7 "archive/zip" 7 "archive/zip"
8 "crypto/sha256" 8 "crypto/sha256"
9 "encoding/base64" 9 "encoding/base64"
10 "encoding/hex" 10 "encoding/hex"
11 "encoding/json" 11 "encoding/json"
12 "fmt" 12 "fmt"
13 "io" 13 "io"
14 "io/ioutil" 14 "io/ioutil"
15 "mime" 15 "mime"
16 "net/http" 16 "net/http"
17 "net/url" 17 "net/url"
18 "os" 18 "os"
19 "path"
19 "path/filepath" 20 "path/filepath"
21 "sort"
20 "strconv" 22 "strconv"
21 "strings" 23 "strings"
22 24
23 "github.com/errgo/errgo" 25 "github.com/errgo/errgo"
24 26
25 "launchpad.net/juju-core/charm" 27 "launchpad.net/juju-core/charm"
26 envtesting "launchpad.net/juju-core/environs/testing" 28 envtesting "launchpad.net/juju-core/environs/testing"
27 "launchpad.net/juju-core/names" 29 "launchpad.net/juju-core/names"
28 "launchpad.net/juju-core/state" 30 "launchpad.net/juju-core/state"
29 "launchpad.net/juju-core/state/api/params" 31 "launchpad.net/juju-core/state/api/params"
30 "launchpad.net/juju-core/state/apiserver/common" 32 "launchpad.net/juju-core/state/apiserver/common"
33 ziputil "launchpad.net/juju-core/utils/zip"
31 ) 34 )
32 35
33 // charmsHandler handles charm upload through HTTPS in the API server. 36 // charmsHandler handles charm upload through HTTPS in the API server.
34 type charmsHandler struct { 37 type charmsHandler struct {
35 state *state.State 38 state *state.State
36 dataDir string 39 dataDir string
37 } 40 }
38 41
39 // zipContentsSenderFunc functions are responsible of sending a zip archive 42 // bundleContentSenderFunc functions are responsible for sending a
40 // related response. The zip archive can be accessed through the given reader. 43 // response related to a charm bundle.
41 type zipContentsSenderFunc func(w http.ResponseWriter, r *http.Request, reader * zip.ReadCloser) 44 type bundleContentSenderFunc func(w http.ResponseWriter, r *http.Request, bundle *charm.Bundle)
42 45
43 func (h *charmsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { 46 func (h *charmsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
44 if err := h.authenticate(r); err != nil { 47 if err := h.authenticate(r); err != nil {
45 h.authError(w) 48 h.authError(w)
46 return 49 return
47 } 50 }
48 51
49 switch r.Method { 52 switch r.Method {
50 case "POST": 53 case "POST":
51 // Add a local charm to the store provider. 54 // Add a local charm to the store provider.
52 // Requires a "series" query specifying the series to use for th e charm. 55 // Requires a "series" query specifying the series to use for th e charm.
53 charmURL, err := h.processPost(r) 56 charmURL, err := h.processPost(r)
54 if err != nil { 57 if err != nil {
55 h.sendError(w, http.StatusBadRequest, err.Error()) 58 h.sendError(w, http.StatusBadRequest, err.Error())
56 return 59 return
57 } 60 }
58 h.sendJSON(w, http.StatusOK, &params.CharmsResponse{CharmURL: ch armURL.String()}) 61 h.sendJSON(w, http.StatusOK, &params.CharmsResponse{CharmURL: ch armURL.String()})
59 case "GET": 62 case "GET":
60 // Retrieve or list charm files. 63 // Retrieve or list charm files.
61 // Requires "url" (charm URL) and an optional "file" (the path t o the 64 // Requires "url" (charm URL) and an optional "file" (the path t o the
62 // charm file) to be included in the query. 65 // charm file) to be included in the query.
63 if charmArchivePath, filePath, err := h.processGet(r); err != ni l { 66 if charmArchivePath, filePath, err := h.processGet(r); err != ni l {
64 // An error occurred retrieving the charm bundle. 67 // An error occurred retrieving the charm bundle.
65 h.sendError(w, http.StatusBadRequest, err.Error()) 68 h.sendError(w, http.StatusBadRequest, err.Error())
66 } else if filePath == "" { 69 } else if filePath == "" {
67 // The client requested the list of charm files. 70 // The client requested the list of charm files.
68 » » » sendZipContents(w, r, charmArchivePath, h.fileListSender ) 71 » » » sendBundleContent(w, r, charmArchivePath, h.manifestSend er)
69 } else { 72 } else {
70 // The client requested a specific file. 73 // The client requested a specific file.
71 » » » sendZipContents(w, r, charmArchivePath, h.fileSender(fil ePath)) 74 » » » sendBundleContent(w, r, charmArchivePath, h.fileSender(f ilePath))
72 } 75 }
73 default: 76 default:
74 h.sendError(w, http.StatusMethodNotAllowed, fmt.Sprintf("unsuppo rted method: %q", r.Method)) 77 h.sendError(w, http.StatusMethodNotAllowed, fmt.Sprintf("unsuppo rted method: %q", r.Method))
75 } 78 }
76 } 79 }
77 80
78 // sendJSON sends a JSON-encoded response to the client. 81 // sendJSON sends a JSON-encoded response to the client.
79 func (h *charmsHandler) sendJSON(w http.ResponseWriter, statusCode int, response *params.CharmsResponse) error { 82 func (h *charmsHandler) sendJSON(w http.ResponseWriter, statusCode int, response *params.CharmsResponse) error {
80 w.Header().Set("Content-Type", "application/json") 83 w.Header().Set("Content-Type", "application/json")
81 w.WriteHeader(statusCode) 84 w.WriteHeader(statusCode)
82 body, err := json.Marshal(response) 85 body, err := json.Marshal(response)
83 if err != nil { 86 if err != nil {
84 return err 87 return err
85 } 88 }
86 w.Write(body) 89 w.Write(body)
87 return nil 90 return nil
88 } 91 }
89 92
90 // sendZipContents uses the given zipContentsSenderFunc to send a response 93 // sendBundleContent uses the given bundleContentSenderFunc to send a response
91 // related to the zip archive located in the given archivePath. 94 // related to the charm archive located in the given archivePath.
92 func sendZipContents(w http.ResponseWriter, r *http.Request, archivePath string, zipContentsSender zipContentsSenderFunc) { 95 func sendBundleContent(w http.ResponseWriter, r *http.Request, archivePath strin g, sender bundleContentSenderFunc) {
93 » reader, err := zip.OpenReader(archivePath) 96 » bundle, err := charm.ReadBundle(archivePath)
94 if err != nil { 97 if err != nil {
95 http.Error( 98 http.Error(
96 w, fmt.Sprintf("unable to read archive in %q: %v", archi vePath, err), 99 w, fmt.Sprintf("unable to read archive in %q: %v", archi vePath, err),
97 http.StatusInternalServerError) 100 http.StatusInternalServerError)
98 return 101 return
99 } 102 }
100 » defer reader.Close() 103 » // The bundleContentSenderFunc will set up and send an appropriate respo nse.
101 » // The zipContentsSenderFunc will handle the zip contents, set up and se nd 104 » sender(w, r, bundle)
102 » // an appropriate response.
103 » zipContentsSender(w, r, reader)
104 } 105 }
105 106
106 // fileListSender sends a JSON-encoded response to the client including the 107 // manifestSender sends a JSON-encoded response to the client including the
107 // list of files contained in the zip archive. 108 // list of files contained in the charm bundle.
108 func (h *charmsHandler) fileListSender(w http.ResponseWriter, r *http.Request, r eader *zip.ReadCloser) { 109 func (h *charmsHandler) manifestSender(w http.ResponseWriter, r *http.Request, b undle *charm.Bundle) {
109 » var files []string 110 » manifest, err := bundle.Manifest()
110 » for _, file := range reader.File { 111 » if err != nil {
111 » » fileInfo := file.FileInfo() 112 » » http.Error(
112 » » if !fileInfo.IsDir() { 113 » » » w, fmt.Sprintf("unable to read archive in %q: %v", bundl e.Path, err),
113 » » » files = append(files, file.Name) 114 » » » http.StatusInternalServerError)
114 » » } 115 » » return
115 } 116 }
116 » h.sendJSON(w, http.StatusOK, &params.CharmsResponse{Files: files}) 117 » h.sendJSON(w, http.StatusOK, &params.CharmsResponse{Files: manifest.Sort edValues()})
117 } 118 }
118 119
119 // fileSender returns a zipContentsSenderFunc which is responsible of sending 120 // fileSender returns a bundleContentSenderFunc which is responsible for sending
120 // the contents of filePath included in the given zip. 121 // the contents of filePath included in the given charm bundle. If filePath does
121 // A 404 page not found is returned if path does not exist in the zip. 122 // not identify a file, or a symlink that resolves to a file, a 403 forbidden er ror
fwereade 2014/03/06 17:43:48 ehh fix docs.
fwereade 2014/03/06 18:12:10 Done.
122 // A 403 forbidden error is returned if path points to a directory. 123 // is returned.
123 func (h *charmsHandler) fileSender(filePath string) zipContentsSenderFunc { 124 func (h *charmsHandler) fileSender(filePath string) bundleContentSenderFunc {
124 » return func(w http.ResponseWriter, r *http.Request, reader *zip.ReadClos er) { 125 » return func(w http.ResponseWriter, r *http.Request, bundle *charm.Bundle ) {
125 » » for _, file := range reader.File { 126 » » // TODO(fwereade) 20140127 lp:1285685
dimitern 2014/03/06 17:54:51 Shouldn't this be // TODO(fwereade) 2014-01-27 bug
fwereade 2014/03/06 18:12:10 Done.
126 » » » if h.fixPath(file.Name) != filePath { 127 » » // This doesn't handle symlinks helpfully, and should be talking in
128 » » // terms of bundles rather than zip readers; but this demands th ought
129 » » // and design and is not amenable to a quick fix.
130 » » zipReader, err := zip.OpenReader(bundle.Path)
131 » » if err != nil {
132 » » » http.Error(
133 » » » » w, fmt.Sprintf("unable to read charm: %v", err),
134 » » » » http.StatusInternalServerError)
135 » » » return
136 » » }
137 » » defer zipReader.Close()
138 » » for _, file := range zipReader.File {
139 » » » if path.Clean(file.Name) != filePath {
127 continue 140 continue
128 } 141 }
129 fileInfo := file.FileInfo() 142 fileInfo := file.FileInfo()
130 if fileInfo.IsDir() { 143 if fileInfo.IsDir() {
131 http.Error(w, "directory listing not allowed", h ttp.StatusForbidden) 144 http.Error(w, "directory listing not allowed", h ttp.StatusForbidden)
132 return 145 return
133 } 146 }
134 » » » if contents, err := file.Open(); err != nil { 147 » » » contents, err := file.Open()
148 » » » if err != nil {
135 http.Error( 149 http.Error(
136 w, fmt.Sprintf("unable to read file %q: %v", filePath, err), 150 w, fmt.Sprintf("unable to read file %q: %v", filePath, err),
137 http.StatusInternalServerError) 151 http.StatusInternalServerError)
138 return 152 return
139 } else {
140 defer contents.Close()
141 ctype := mime.TypeByExtension(filepath.Ext(fileP ath))
142 if ctype != "" {
143 w.Header().Set("Content-Type", ctype)
144 }
145 w.Header().Set("Content-Length", strconv.FormatI nt(fileInfo.Size(), 10))
146 w.WriteHeader(http.StatusOK)
147 io.Copy(w, contents)
148 } 153 }
154 defer contents.Close()
155 ctype := mime.TypeByExtension(filepath.Ext(filePath))
156 if ctype != "" {
157 w.Header().Set("Content-Type", ctype)
158 }
159 w.Header().Set("Content-Length", strconv.FormatInt(fileI nfo.Size(), 10))
160 w.WriteHeader(http.StatusOK)
161 io.Copy(w, contents)
149 return 162 return
150 } 163 }
151 http.NotFound(w, r) 164 http.NotFound(w, r)
152 return 165 return
153 } 166 }
154 } 167 }
155 168
156 // sendError sends a JSON-encoded error response. 169 // sendError sends a JSON-encoded error response.
157 func (h *charmsHandler) sendError(w http.ResponseWriter, statusCode int, message string) error { 170 func (h *charmsHandler) sendError(w http.ResponseWriter, statusCode int, message string) error {
158 return h.sendJSON(w, statusCode, &params.CharmsResponse{Error: message}) 171 return h.sendJSON(w, statusCode, &params.CharmsResponse{Error: message})
(...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after
266 zipr, err := zip.NewReader(f, fi.Size()) 279 zipr, err := zip.NewReader(f, fi.Size())
267 if err != nil { 280 if err != nil {
268 return errgo.Annotate(err, "cannot open charm archive") 281 return errgo.Annotate(err, "cannot open charm archive")
269 } 282 }
270 283
271 // Find out the root dir prefix from the archive. 284 // Find out the root dir prefix from the archive.
272 rootDir, err := h.findArchiveRootDir(zipr) 285 rootDir, err := h.findArchiveRootDir(zipr)
273 if err != nil { 286 if err != nil {
274 return errgo.Annotate(err, "cannot read charm archive") 287 return errgo.Annotate(err, "cannot read charm archive")
275 } 288 }
276 » if rootDir == "" { 289 » if rootDir == "." {
277 // Normal charm, just use charm.ReadBundle(). 290 // Normal charm, just use charm.ReadBundle().
278 return nil 291 return nil
279 } 292 }
293
280 // There is one or more subdirs, so we need extract it to a temp 294 // There is one or more subdirs, so we need extract it to a temp
281 » // dir and then read is as a charm dir. 295 » // dir and then read it as a charm dir.
282 tempDir, err := ioutil.TempDir("", "charm-extract") 296 tempDir, err := ioutil.TempDir("", "charm-extract")
283 if err != nil { 297 if err != nil {
284 return errgo.Annotate(err, "cannot create temp directory") 298 return errgo.Annotate(err, "cannot create temp directory")
285 } 299 }
286 defer os.RemoveAll(tempDir) 300 defer os.RemoveAll(tempDir)
287 » err = h.extractArchiveTo(zipr, rootDir, tempDir) 301 » if err := ziputil.Extract(zipr, tempDir, rootDir); err != nil {
288 » if err != nil {
289 return errgo.Annotate(err, "cannot extract charm archive") 302 return errgo.Annotate(err, "cannot extract charm archive")
290 } 303 }
291 dir, err := charm.ReadDir(tempDir) 304 dir, err := charm.ReadDir(tempDir)
292 if err != nil { 305 if err != nil {
293 return errgo.Annotate(err, "cannot read extracted archive") 306 return errgo.Annotate(err, "cannot read extracted archive")
294 } 307 }
308
295 // Now repackage the dir as a bundle at the original path. 309 // Now repackage the dir as a bundle at the original path.
296 if err := f.Truncate(0); err != nil { 310 if err := f.Truncate(0); err != nil {
297 return err 311 return err
298 } 312 }
299 if err := dir.BundleTo(f); err != nil { 313 if err := dir.BundleTo(f); err != nil {
300 return err 314 return err
301 } 315 }
302 return nil 316 return nil
303 } 317 }
304 318
305 // fixPath converts all forward and backslashes in path to the OS path
306 // separator and calls filepath.Clean before returning it.
307 func (h *charmsHandler) fixPath(path string) string {
308 sep := string(filepath.Separator)
309 p := strings.Replace(path, "\\", sep, -1)
310 return filepath.Clean(strings.Replace(p, "/", sep, -1))
311 }
312
313 // findArchiveRootDir scans a zip archive and returns the rootDir of 319 // findArchiveRootDir scans a zip archive and returns the rootDir of
314 // the archive, the one containing metadata.yaml, config.yaml and 320 // the archive, the one containing metadata.yaml, config.yaml and
315 // revision files, or an error if the archive appears invalid. 321 // revision files, or an error if the archive appears invalid.
316 func (h *charmsHandler) findArchiveRootDir(zipr *zip.Reader) (string, error) { 322 func (h *charmsHandler) findArchiveRootDir(zipr *zip.Reader) (string, error) {
317 » numFound := 0 323 » paths, err := ziputil.Find(zipr, "metadata.yaml")
318 » metadataFound := false // metadata.yaml is the only required file. 324 » if err != nil {
319 » rootPath := "" 325 » » return "", err
320 » lookFor := []string{"metadata.yaml", "config.yaml", "revision"} 326 » }
321 » for _, fh := range zipr.File { 327 » switch len(paths) {
322 » » for _, fname := range lookFor { 328 » case 0:
323 » » » dir, file := filepath.Split(h.fixPath(fh.Name)) 329 » » return "", fmt.Errorf("invalid charm archive: missing metadata.y aml")
324 » » » if file == fname { 330 » case 1:
325 » » » » if file == "metadata.yaml" { 331 » default:
326 » » » » » metadataFound = true 332 » » sort.Sort(byDepth(paths))
327 » » » » } 333 » » if depth(paths[0]) == depth(paths[1]) {
328 » » » » numFound++ 334 » » » return "", fmt.Errorf("invalid charm archive: ambiguous root directory")
329 » » » » if rootPath == "" {
330 » » » » » rootPath = dir
331 » » » » } else if rootPath != dir {
332 » » » » » return "", fmt.Errorf("invalid charm arc hive: expected all %v files in the same directory", lookFor)
333 » » » » }
334 » » » » if numFound == len(lookFor) {
335 » » » » » return rootPath, nil
336 » » » » }
337 » » » }
338 } 335 }
339 } 336 }
340 » if !metadataFound { 337 » return filepath.Dir(paths[0]), nil
341 » » return "", fmt.Errorf("invalid charm archive: missing metadata.y aml")
342 » }
343 » return rootPath, nil
344 } 338 }
345 339
346 // extractArchiveTo extracts an archive to the given destDir, removing 340 func depth(path string) int {
347 // the rootDir from each file, effectively reducing any nested subdirs 341 » return strings.Count(path, string(filepath.Separator))
dimitern 2014/03/06 17:54:51 return strings.Count(path, "/") ? how about window
fwereade 2014/03/06 18:12:10 dammit, yeah, that code used to run against the fi
348 // to the root level.
349 func (h *charmsHandler) extractArchiveTo(zipr *zip.Reader, rootDir, destDir stri ng) error {
350 » for _, fh := range zipr.File {
351 » » err := h.extractSingleFile(fh, rootDir, destDir)
352 » » if err != nil {
353 » » » return err
354 » » }
355 » }
356 » return nil
357 } 342 }
358 343
359 // extractSingleFile extracts the given zip file header, removing 344 type byDepth []string
360 // rootDir from the filename, to the destDir.
361 func (h *charmsHandler) extractSingleFile(fh *zip.File, rootDir, destDir string) error {
362 » cleanName := h.fixPath(fh.Name)
363 » relName, err := filepath.Rel(rootDir, cleanName)
364 » if err != nil {
365 » » // Skip paths not relative to roo
366 » » return nil
367 » }
368 » if strings.Contains(relName, "..") || relName == "." {
369 » » // Skip current dir and paths outside rootDir.
370 » » return nil
371 » }
372 » dirName := filepath.Dir(relName)
373 » f, err := fh.Open()
374 » if err != nil {
375 » » return err
376 » }
377 » defer f.Close()
378 345
379 » mode := fh.Mode() 346 func (d byDepth) Len() int { return len(d) }
380 » destPath := filepath.Join(destDir, relName) 347 func (d byDepth) Swap(i, j int) { d[i], d[j] = d[j], d[i] }
381 » if dirName != "" && mode&os.ModeDir != 0 { 348 func (d byDepth) Less(i, j int) bool { return depth(d[i]) < depth(d[j]) }
382 » » err = os.MkdirAll(destPath, mode&0777)
383 » » if err != nil {
384 » » » return err
385 » » }
386 » » return nil
387 » }
388
389 » if mode&os.ModeSymlink != 0 {
390 » » data, err := ioutil.ReadAll(f)
391 » » if err != nil {
392 » » » return err
393 » » }
394 » » target := string(data)
395 » » if filepath.IsAbs(target) {
396 » » » return fmt.Errorf("symlink %q is absolute: %q", cleanNam e, target)
397 » » }
398 » » p := filepath.Join(dirName, target)
399 » » if strings.Contains(p, "..") {
400 » » » return fmt.Errorf("symlink %q links out of charm: %s", c leanName, target)
401 » » }
402 » » err = os.Symlink(target, destPath)
403 » » if err != nil {
404 » » » return err
405 » » }
406 » }
407 » if dirName == "hooks" {
408 » » if mode&os.ModeType == 0 {
409 » » » // Set all hooks executable (by owner)
410 » » » mode = mode | 0100
411 » » }
412 » }
413
414 » // Check file type.
415 » e := "file has an unknown type: %q"
416 » switch mode & os.ModeType {
417 » case os.ModeDir, os.ModeSymlink, 0:
418 » » // That's expected, it's ok.
419 » » e = ""
420 » case os.ModeNamedPipe:
421 » » e = "file is a named pipe: %q"
422 » case os.ModeSocket:
423 » » e = "file is a socket: %q"
424 » case os.ModeDevice:
425 » » e = "file is a device: %q"
426 » }
427 » if e != "" {
428 » » return fmt.Errorf(e, destPath)
429 » }
430
431 » out, err := os.OpenFile(destPath, os.O_CREATE|os.O_WRONLY, mode&0777)
432 » if err != nil {
433 » » return fmt.Errorf("creating %q failed: %v", destPath, err)
434 » }
435 » defer out.Close()
436 » _, err = io.Copy(out, f)
437 » return err
438 }
439 349
440 // repackageAndUploadCharm expands the given charm archive to a 350 // repackageAndUploadCharm expands the given charm archive to a
441 // temporary directoy, repackages it with the given curl's revision, 351 // temporary directoy, repackages it with the given curl's revision,
442 // then uploads it to providr storage, and finally updates the state. 352 // then uploads it to providr storage, and finally updates the state.
443 func (h *charmsHandler) repackageAndUploadCharm(archive *charm.Bundle, curl *cha rm.URL) error { 353 func (h *charmsHandler) repackageAndUploadCharm(archive *charm.Bundle, curl *cha rm.URL) error {
444 // Create a temp dir to contain the extracted charm 354 // Create a temp dir to contain the extracted charm
445 // dir and the repackaged archive. 355 // dir and the repackaged archive.
446 tempDir, err := ioutil.TempDir("", "charm-download") 356 tempDir, err := ioutil.TempDir("", "charm-download")
447 if err != nil { 357 if err != nil {
448 return errgo.Annotate(err, "cannot create temp directory") 358 return errgo.Annotate(err, "cannot create temp directory")
449 } 359 }
450 defer os.RemoveAll(tempDir) 360 defer os.RemoveAll(tempDir)
451 extractPath := filepath.Join(tempDir, "extracted") 361 extractPath := filepath.Join(tempDir, "extracted")
452 repackagedPath := filepath.Join(tempDir, "repackaged.zip") 362 repackagedPath := filepath.Join(tempDir, "repackaged.zip")
453 repackagedArchive, err := os.Create(repackagedPath) 363 repackagedArchive, err := os.Create(repackagedPath)
454 if err != nil { 364 if err != nil {
455 return errgo.Annotate(err, "cannot repackage uploaded charm") 365 return errgo.Annotate(err, "cannot repackage uploaded charm")
456 } 366 }
457 defer repackagedArchive.Close() 367 defer repackagedArchive.Close()
458 368
459 // Expand and repack it with the revision specified by curl. 369 // Expand and repack it with the revision specified by curl.
460 archive.SetRevision(curl.Revision) 370 archive.SetRevision(curl.Revision)
461 if err := archive.ExpandTo(extractPath); err != nil { 371 if err := archive.ExpandTo(extractPath); err != nil {
462 return errgo.Annotate(err, "cannot extract uploaded charm") 372 return errgo.Annotate(err, "cannot extract uploaded charm")
463 } 373 }
464 charmDir, err := charm.ReadDir(extractPath) 374 charmDir, err := charm.ReadDir(extractPath)
465 if err != nil { 375 if err != nil {
466 return errgo.Annotate(err, "cannot read extracted charm") 376 return errgo.Annotate(err, "cannot read extracted charm")
467 } 377 }
378
468 // Bundle the charm and calculate its sha256 hash at the 379 // Bundle the charm and calculate its sha256 hash at the
469 // same time. 380 // same time.
470 hash := sha256.New() 381 hash := sha256.New()
471 err = charmDir.BundleTo(io.MultiWriter(hash, repackagedArchive)) 382 err = charmDir.BundleTo(io.MultiWriter(hash, repackagedArchive))
472 if err != nil { 383 if err != nil {
473 return errgo.Annotate(err, "cannot repackage uploaded charm") 384 return errgo.Annotate(err, "cannot repackage uploaded charm")
474 } 385 }
475 bundleSHA256 := hex.EncodeToString(hash.Sum(nil)) 386 bundleSHA256 := hex.EncodeToString(hash.Sum(nil))
476 size, err := repackagedArchive.Seek(0, 2) 387 size, err := repackagedArchive.Seek(0, 2)
477 if err != nil { 388 if err != nil {
478 return errgo.Annotate(err, "cannot get charm file size") 389 return errgo.Annotate(err, "cannot get charm file size")
479 } 390 }
480 » // Seek to the beginning so the subsequent Put will read 391
481 » // the whole file again. 392 » // Now upload to provider storage.
482 if _, err := repackagedArchive.Seek(0, 0); err != nil { 393 if _, err := repackagedArchive.Seek(0, 0); err != nil {
483 return errgo.Annotate(err, "cannot rewind the charm file reader" ) 394 return errgo.Annotate(err, "cannot rewind the charm file reader" )
484 } 395 }
485
486 // Now upload to provider storage.
487 storage, err := envtesting.GetEnvironStorage(h.state) 396 storage, err := envtesting.GetEnvironStorage(h.state)
488 if err != nil { 397 if err != nil {
489 return errgo.Annotate(err, "cannot access provider storage") 398 return errgo.Annotate(err, "cannot access provider storage")
490 } 399 }
491 name := charm.Quote(curl.String()) 400 name := charm.Quote(curl.String())
492 if err := storage.Put(name, repackagedArchive, size); err != nil { 401 if err := storage.Put(name, repackagedArchive, size); err != nil {
493 return errgo.Annotate(err, "cannot upload charm to provider stor age") 402 return errgo.Annotate(err, "cannot upload charm to provider stor age")
494 } 403 }
495 storageURL, err := storage.URL(name) 404 storageURL, err := storage.URL(name)
496 if err != nil { 405 if err != nil {
(...skipping 20 matching lines...) Expand all
517 // Retrieve and validate query parameters. 426 // Retrieve and validate query parameters.
518 curl := query.Get("url") 427 curl := query.Get("url")
519 if curl == "" { 428 if curl == "" {
520 return "", "", fmt.Errorf("expected url=CharmURL query argument" ) 429 return "", "", fmt.Errorf("expected url=CharmURL query argument" )
521 } 430 }
522 var filePath string 431 var filePath string
523 file := query.Get("file") 432 file := query.Get("file")
524 if file == "" { 433 if file == "" {
525 filePath = "" 434 filePath = ""
526 } else { 435 } else {
527 » » filePath = h.fixPath(file) 436 » » filePath = path.Clean(file)
528 } 437 }
529 438
530 // Prepare the bundle directories. 439 // Prepare the bundle directories.
531 name := charm.Quote(curl) 440 name := charm.Quote(curl)
532 charmArchivePath := filepath.Join(h.dataDir, "charm-get-cache", name+".z ip") 441 charmArchivePath := filepath.Join(h.dataDir, "charm-get-cache", name+".z ip")
533 442
534 // Check if the charm archive is already in the cache. 443 // Check if the charm archive is already in the cache.
535 if _, err := os.Stat(charmArchivePath); os.IsNotExist(err) { 444 if _, err := os.Stat(charmArchivePath); os.IsNotExist(err) {
536 // Download the charm archive and save it to the cache. 445 // Download the charm archive and save it to the cache.
537 if err = h.downloadCharm(name, charmArchivePath); err != nil { 446 if err = h.downloadCharm(name, charmArchivePath); err != nil {
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
577 defer tempCharmArchive.Close() 486 defer tempCharmArchive.Close()
578 if err = ioutil.WriteFile(tempCharmArchive.Name(), data, 0644); err != n il { 487 if err = ioutil.WriteFile(tempCharmArchive.Name(), data, 0644); err != n il {
579 return errgo.Annotate(err, "error processing charm archive downl oad") 488 return errgo.Annotate(err, "error processing charm archive downl oad")
580 } 489 }
581 if err = os.Rename(tempCharmArchive.Name(), charmArchivePath); err != ni l { 490 if err = os.Rename(tempCharmArchive.Name(), charmArchivePath); err != ni l {
582 defer os.Remove(tempCharmArchive.Name()) 491 defer os.Remove(tempCharmArchive.Name())
583 return errgo.Annotate(err, "error renaming the charm archive") 492 return errgo.Annotate(err, "error renaming the charm archive")
584 } 493 }
585 return nil 494 return nil
586 } 495 }
OLDNEW

Powered by Google App Engine
RSS Feeds Recent Issues | This issue
This is Rietveld f62528b