Left: | ||
Right: |
OLD | NEW |
---|---|
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, ¶ms.CharmsResponse{CharmURL: ch armURL.String()}) | 61 h.sendJSON(w, http.StatusOK, ¶ms.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, ¶ms.CharmsResponse{Files: files}) | 117 » h.sendJSON(w, http.StatusOK, ¶ms.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, ¶ms.CharmsResponse{Error: message}) | 171 return h.sendJSON(w, statusCode, ¶ms.CharmsResponse{Error: message}) |
(...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
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 Loading... | |
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 Loading... | |
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 } |
OLD | NEW |