blob: 63fb854b3117877d0f99624bab8322f077cd97d9 [file] [log] [blame]
// Copyright 2016 Canonical Ltd.
// Licensed under the LGPLv3, see LICENCE file for details.
package gomaasapi
import (
"encoding/base64"
"net/http"
"net/url"
"github.com/juju/errors"
"github.com/juju/schema"
"github.com/juju/version"
)
type file struct {
controller *controller
resourceURI string
filename string
anonymousURI *url.URL
content string
}
// Filename implements File.
func (f *file) Filename() string {
return f.filename
}
// AnonymousURL implements File.
func (f *file) AnonymousURL() string {
url := f.controller.client.GetURL(f.anonymousURI)
return url.String()
}
// Delete implements File.
func (f *file) Delete() error {
err := f.controller.delete(f.resourceURI)
if err != nil {
if svrErr, ok := errors.Cause(err).(ServerError); ok {
switch svrErr.StatusCode {
case http.StatusNotFound:
return errors.Wrap(err, NewNoMatchError(svrErr.BodyMessage))
case http.StatusForbidden:
return errors.Wrap(err, NewPermissionError(svrErr.BodyMessage))
}
}
return NewUnexpectedError(err)
}
return nil
}
// ReadAll implements File.
func (f *file) ReadAll() ([]byte, error) {
if f.content == "" {
return f.readFromServer()
}
bytes, err := base64.StdEncoding.DecodeString(f.content)
if err != nil {
return nil, NewUnexpectedError(err)
}
return bytes, nil
}
func (f *file) readFromServer() ([]byte, error) {
// If the content is available, it is base64 encoded, so
args := make(url.Values)
args.Add("filename", f.filename)
bytes, err := f.controller._getRaw("files", "get", args)
if err != nil {
if svrErr, ok := errors.Cause(err).(ServerError); ok {
switch svrErr.StatusCode {
case http.StatusNotFound:
return nil, errors.Wrap(err, NewNoMatchError(svrErr.BodyMessage))
case http.StatusForbidden:
return nil, errors.Wrap(err, NewPermissionError(svrErr.BodyMessage))
}
}
return nil, NewUnexpectedError(err)
}
return bytes, nil
}
func readFiles(controllerVersion version.Number, source interface{}) ([]*file, error) {
readFunc, err := getFileDeserializationFunc(controllerVersion)
if err != nil {
return nil, errors.Trace(err)
}
checker := schema.List(schema.StringMap(schema.Any()))
coerced, err := checker.Coerce(source, nil)
if err != nil {
return nil, WrapWithDeserializationError(err, "file base schema check failed")
}
valid := coerced.([]interface{})
return readFileList(valid, readFunc)
}
func readFile(controllerVersion version.Number, source interface{}) (*file, error) {
readFunc, err := getFileDeserializationFunc(controllerVersion)
if err != nil {
return nil, errors.Trace(err)
}
checker := schema.StringMap(schema.Any())
coerced, err := checker.Coerce(source, nil)
if err != nil {
return nil, WrapWithDeserializationError(err, "file base schema check failed")
}
valid := coerced.(map[string]interface{})
return readFunc(valid)
}
func getFileDeserializationFunc(controllerVersion version.Number) (fileDeserializationFunc, error) {
var deserialisationVersion version.Number
for v := range fileDeserializationFuncs {
if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 {
deserialisationVersion = v
}
}
if deserialisationVersion == version.Zero {
return nil, NewUnsupportedVersionError("no file read func for version %s", controllerVersion)
}
return fileDeserializationFuncs[deserialisationVersion], nil
}
// readFileList expects the values of the sourceList to be string maps.
func readFileList(sourceList []interface{}, readFunc fileDeserializationFunc) ([]*file, error) {
result := make([]*file, 0, len(sourceList))
for i, value := range sourceList {
source, ok := value.(map[string]interface{})
if !ok {
return nil, NewDeserializationError("unexpected value for file %d, %T", i, value)
}
file, err := readFunc(source)
if err != nil {
return nil, errors.Annotatef(err, "file %d", i)
}
result = append(result, file)
}
return result, nil
}
type fileDeserializationFunc func(map[string]interface{}) (*file, error)
var fileDeserializationFuncs = map[version.Number]fileDeserializationFunc{
twoDotOh: file_2_0,
}
func file_2_0(source map[string]interface{}) (*file, error) {
fields := schema.Fields{
"resource_uri": schema.String(),
"filename": schema.String(),
"anon_resource_uri": schema.String(),
"content": schema.String(),
}
defaults := schema.Defaults{
"content": "",
}
checker := schema.FieldMap(fields, defaults)
coerced, err := checker.Coerce(source, nil)
if err != nil {
return nil, WrapWithDeserializationError(err, "file 2.0 schema check failed")
}
valid := coerced.(map[string]interface{})
// From here we know that the map returned from the schema coercion
// contains fields of the right type.
anonURI, err := url.ParseRequestURI(valid["anon_resource_uri"].(string))
if err != nil {
return nil, NewUnexpectedError(err)
}
result := &file{
resourceURI: valid["resource_uri"].(string),
filename: valid["filename"].(string),
anonymousURI: anonURI,
content: valid["content"].(string),
}
return result, nil
}