session_v2.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. package registry
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "io"
  6. "io/ioutil"
  7. "net/url"
  8. "strconv"
  9. log "github.com/Sirupsen/logrus"
  10. "github.com/docker/docker/utils"
  11. "github.com/gorilla/mux"
  12. )
  13. func newV2RegistryRouter() *mux.Router {
  14. router := mux.NewRouter()
  15. v2Router := router.PathPrefix("/v2/").Subrouter()
  16. // Version Info
  17. v2Router.Path("/version").Name("version")
  18. // Image Manifests
  19. v2Router.Path("/manifest/{imagename:[a-z0-9-._/]+}/{tagname:[a-zA-Z0-9-._]+}").Name("manifests")
  20. // List Image Tags
  21. v2Router.Path("/tags/{imagename:[a-z0-9-._/]+}").Name("tags")
  22. // Download a blob
  23. v2Router.Path("/blob/{imagename:[a-z0-9-._/]+}/{sumtype:[a-z0-9._+-]+}/{sum:[a-fA-F0-9]{4,}}").Name("downloadBlob")
  24. // Upload a blob
  25. v2Router.Path("/blob/{imagename:[a-z0-9-._/]+}/{sumtype:[a-z0-9._+-]+}").Name("uploadBlob")
  26. // Mounting a blob in an image
  27. v2Router.Path("/mountblob/{imagename:[a-z0-9-._/]+}/{sumtype:[a-z0-9._+-]+}/{sum:[a-fA-F0-9]{4,}}").Name("mountBlob")
  28. return router
  29. }
  30. // APIVersion2 /v2/
  31. var v2HTTPRoutes = newV2RegistryRouter()
  32. func getV2URL(e *Endpoint, routeName string, vars map[string]string) (*url.URL, error) {
  33. route := v2HTTPRoutes.Get(routeName)
  34. if route == nil {
  35. return nil, fmt.Errorf("unknown regisry v2 route name: %q", routeName)
  36. }
  37. varReplace := make([]string, 0, len(vars)*2)
  38. for key, val := range vars {
  39. varReplace = append(varReplace, key, val)
  40. }
  41. routePath, err := route.URLPath(varReplace...)
  42. if err != nil {
  43. return nil, fmt.Errorf("unable to make registry route %q with vars %v: %s", routeName, vars, err)
  44. }
  45. u, err := url.Parse(REGISTRYSERVER)
  46. if err != nil {
  47. return nil, fmt.Errorf("invalid registry url: %s", err)
  48. }
  49. return &url.URL{
  50. Scheme: u.Scheme,
  51. Host: u.Host,
  52. Path: routePath.Path,
  53. }, nil
  54. }
  55. // V2 Provenance POC
  56. func (r *Session) GetV2Version(token []string) (*RegistryInfo, error) {
  57. routeURL, err := getV2URL(r.indexEndpoint, "version", nil)
  58. if err != nil {
  59. return nil, err
  60. }
  61. method := "GET"
  62. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  63. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  64. if err != nil {
  65. return nil, err
  66. }
  67. setTokenAuth(req, token)
  68. res, _, err := r.doRequest(req)
  69. if err != nil {
  70. return nil, err
  71. }
  72. defer res.Body.Close()
  73. if res.StatusCode != 200 {
  74. return nil, utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d fetching Version", res.StatusCode), res)
  75. }
  76. decoder := json.NewDecoder(res.Body)
  77. versionInfo := new(RegistryInfo)
  78. err = decoder.Decode(versionInfo)
  79. if err != nil {
  80. return nil, fmt.Errorf("unable to decode GetV2Version JSON response: %s", err)
  81. }
  82. return versionInfo, nil
  83. }
  84. //
  85. // 1) Check if TarSum of each layer exists /v2/
  86. // 1.a) if 200, continue
  87. // 1.b) if 300, then push the
  88. // 1.c) if anything else, err
  89. // 2) PUT the created/signed manifest
  90. //
  91. func (r *Session) GetV2ImageManifest(imageName, tagName string, token []string) ([]byte, error) {
  92. vars := map[string]string{
  93. "imagename": imageName,
  94. "tagname": tagName,
  95. }
  96. routeURL, err := getV2URL(r.indexEndpoint, "manifests", vars)
  97. if err != nil {
  98. return nil, err
  99. }
  100. method := "GET"
  101. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  102. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  103. if err != nil {
  104. return nil, err
  105. }
  106. setTokenAuth(req, token)
  107. res, _, err := r.doRequest(req)
  108. if err != nil {
  109. return nil, err
  110. }
  111. defer res.Body.Close()
  112. if res.StatusCode != 200 {
  113. if res.StatusCode == 401 {
  114. return nil, errLoginRequired
  115. } else if res.StatusCode == 404 {
  116. return nil, ErrDoesNotExist
  117. }
  118. return nil, utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to fetch for %s:%s", res.StatusCode, imageName, tagName), res)
  119. }
  120. buf, err := ioutil.ReadAll(res.Body)
  121. if err != nil {
  122. return nil, fmt.Errorf("Error while reading the http response: %s", err)
  123. }
  124. return buf, nil
  125. }
  126. // - Succeeded to mount for this image scope
  127. // - Failed with no error (So continue to Push the Blob)
  128. // - Failed with error
  129. func (r *Session) PostV2ImageMountBlob(imageName, sumType, sum string, token []string) (bool, error) {
  130. vars := map[string]string{
  131. "imagename": imageName,
  132. "sumtype": sumType,
  133. "sum": sum,
  134. }
  135. routeURL, err := getV2URL(r.indexEndpoint, "mountBlob", vars)
  136. if err != nil {
  137. return false, err
  138. }
  139. method := "POST"
  140. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  141. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  142. if err != nil {
  143. return false, err
  144. }
  145. setTokenAuth(req, token)
  146. res, _, err := r.doRequest(req)
  147. if err != nil {
  148. return false, err
  149. }
  150. res.Body.Close() // close early, since we're not needing a body on this call .. yet?
  151. switch res.StatusCode {
  152. case 200:
  153. // return something indicating no push needed
  154. return true, nil
  155. case 300:
  156. // return something indicating blob push needed
  157. return false, nil
  158. }
  159. return false, fmt.Errorf("Failed to mount %q - %s:%s : %d", imageName, sumType, sum, res.StatusCode)
  160. }
  161. func (r *Session) GetV2ImageBlob(imageName, sumType, sum string, blobWrtr io.Writer, token []string) error {
  162. vars := map[string]string{
  163. "imagename": imageName,
  164. "sumtype": sumType,
  165. "sum": sum,
  166. }
  167. routeURL, err := getV2URL(r.indexEndpoint, "downloadBlob", vars)
  168. if err != nil {
  169. return err
  170. }
  171. method := "GET"
  172. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  173. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  174. if err != nil {
  175. return err
  176. }
  177. setTokenAuth(req, token)
  178. res, _, err := r.doRequest(req)
  179. if err != nil {
  180. return err
  181. }
  182. defer res.Body.Close()
  183. if res.StatusCode != 200 {
  184. if res.StatusCode == 401 {
  185. return errLoginRequired
  186. }
  187. return utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to pull %s blob", res.StatusCode, imageName), res)
  188. }
  189. _, err = io.Copy(blobWrtr, res.Body)
  190. return err
  191. }
  192. func (r *Session) GetV2ImageBlobReader(imageName, sumType, sum string, token []string) (io.ReadCloser, int64, error) {
  193. vars := map[string]string{
  194. "imagename": imageName,
  195. "sumtype": sumType,
  196. "sum": sum,
  197. }
  198. routeURL, err := getV2URL(r.indexEndpoint, "downloadBlob", vars)
  199. if err != nil {
  200. return nil, 0, err
  201. }
  202. method := "GET"
  203. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  204. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  205. if err != nil {
  206. return nil, 0, err
  207. }
  208. setTokenAuth(req, token)
  209. res, _, err := r.doRequest(req)
  210. if err != nil {
  211. return nil, 0, err
  212. }
  213. if res.StatusCode != 200 {
  214. if res.StatusCode == 401 {
  215. return nil, 0, errLoginRequired
  216. }
  217. return nil, 0, utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to pull %s blob", res.StatusCode, imageName), res)
  218. }
  219. lenStr := res.Header.Get("Content-Length")
  220. l, err := strconv.ParseInt(lenStr, 10, 64)
  221. if err != nil {
  222. return nil, 0, err
  223. }
  224. return res.Body, l, err
  225. }
  226. // Push the image to the server for storage.
  227. // 'layer' is an uncompressed reader of the blob to be pushed.
  228. // The server will generate it's own checksum calculation.
  229. func (r *Session) PutV2ImageBlob(imageName, sumType string, blobRdr io.Reader, token []string) (serverChecksum string, err error) {
  230. vars := map[string]string{
  231. "imagename": imageName,
  232. "sumtype": sumType,
  233. }
  234. routeURL, err := getV2URL(r.indexEndpoint, "uploadBlob", vars)
  235. if err != nil {
  236. return "", err
  237. }
  238. method := "PUT"
  239. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  240. req, err := r.reqFactory.NewRequest(method, routeURL.String(), blobRdr)
  241. if err != nil {
  242. return "", err
  243. }
  244. setTokenAuth(req, token)
  245. res, _, err := r.doRequest(req)
  246. if err != nil {
  247. return "", err
  248. }
  249. defer res.Body.Close()
  250. if res.StatusCode != 201 {
  251. if res.StatusCode == 401 {
  252. return "", errLoginRequired
  253. }
  254. return "", utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to push %s blob", res.StatusCode, imageName), res)
  255. }
  256. type sumReturn struct {
  257. Checksum string `json:"checksum"`
  258. }
  259. decoder := json.NewDecoder(res.Body)
  260. var sumInfo sumReturn
  261. err = decoder.Decode(&sumInfo)
  262. if err != nil {
  263. return "", fmt.Errorf("unable to decode PutV2ImageBlob JSON response: %s", err)
  264. }
  265. // XXX this is a json struct from the registry, with its checksum
  266. return sumInfo.Checksum, nil
  267. }
  268. // Finally Push the (signed) manifest of the blobs we've just pushed
  269. func (r *Session) PutV2ImageManifest(imageName, tagName string, manifestRdr io.Reader, token []string) error {
  270. vars := map[string]string{
  271. "imagename": imageName,
  272. "tagname": tagName,
  273. }
  274. routeURL, err := getV2URL(r.indexEndpoint, "manifests", vars)
  275. if err != nil {
  276. return err
  277. }
  278. method := "PUT"
  279. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  280. req, err := r.reqFactory.NewRequest(method, routeURL.String(), manifestRdr)
  281. if err != nil {
  282. return err
  283. }
  284. setTokenAuth(req, token)
  285. res, _, err := r.doRequest(req)
  286. if err != nil {
  287. return err
  288. }
  289. res.Body.Close()
  290. if res.StatusCode != 201 {
  291. if res.StatusCode == 401 {
  292. return errLoginRequired
  293. }
  294. return utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to push %s:%s manifest", res.StatusCode, imageName, tagName), res)
  295. }
  296. return nil
  297. }
  298. // Given a repository name, returns a json array of string tags
  299. func (r *Session) GetV2RemoteTags(imageName string, token []string) ([]string, error) {
  300. vars := map[string]string{
  301. "imagename": imageName,
  302. }
  303. routeURL, err := getV2URL(r.indexEndpoint, "tags", vars)
  304. if err != nil {
  305. return nil, err
  306. }
  307. method := "GET"
  308. log.Debugf("[registry] Calling %q %s", method, routeURL.String())
  309. req, err := r.reqFactory.NewRequest(method, routeURL.String(), nil)
  310. if err != nil {
  311. return nil, err
  312. }
  313. setTokenAuth(req, token)
  314. res, _, err := r.doRequest(req)
  315. if err != nil {
  316. return nil, err
  317. }
  318. defer res.Body.Close()
  319. if res.StatusCode != 200 {
  320. if res.StatusCode == 401 {
  321. return nil, errLoginRequired
  322. } else if res.StatusCode == 404 {
  323. return nil, ErrDoesNotExist
  324. }
  325. return nil, utils.NewHTTPRequestError(fmt.Sprintf("Server error: %d trying to fetch for %s", res.StatusCode, imageName), res)
  326. }
  327. decoder := json.NewDecoder(res.Body)
  328. var tags []string
  329. err = decoder.Decode(&tags)
  330. if err != nil {
  331. return nil, fmt.Errorf("Error while decoding the http response: %s", err)
  332. }
  333. return tags, nil
  334. }