From c8b964ca4dbf236522e38fe190fbf66347f7b73b Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 24 Jan 2020 14:09:27 -0800 Subject: [PATCH 01/21] Add the complete set of machine state enums Fix NodeStatus enum names --- enum.go | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/enum.go b/enum.go index a516d6b..1aaa218 100644 --- a/enum.go +++ b/enum.go @@ -54,4 +54,25 @@ const ( // The node failed to erase its disks. NodeStatusFailedDiskErasing = "15" + + // The node is in rescue mode. + NodeStatusRescueMode = "16" + + // The node is entering rescue mode. + NodeStatusEnteringRescueMode = "17" + + // The node failed to enter rescue mode. + NodeStatusFailedEnteringRescueMode = "18" + + // The node is exiting rescue mode. + NodeStatusExitingRescueMode = "19" + + // The node failed to exit rescue mode. + NodeStatusFailedExitingRescueMode = "20" + + // Running tests on Node + NodeStatusTesting = "21" + + // Testing has failed + NodeStatusFailedTesting = "22" ) From 976de7f6446a121bfe313584b1e55d70fba737ab Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 21 Jan 2020 19:01:59 -0800 Subject: [PATCH 02/21] Add go module support --- dependencies.tsv | 13 ------------- go.mod | 19 +++++++++++++++++++ go.sum | 19 +++++++++++++++++++ 3 files changed, 38 insertions(+), 13 deletions(-) delete mode 100644 dependencies.tsv create mode 100644 go.mod create mode 100644 go.sum diff --git a/dependencies.tsv b/dependencies.tsv deleted file mode 100644 index e1143b8..0000000 --- a/dependencies.tsv +++ /dev/null @@ -1,13 +0,0 @@ -github.com/juju/collections git 520e0549d51ae2b50f44f4df2b145a780a5bc6e0 2018-05-15T20:37:31Z -github.com/juju/errors git 1b5e39b83d1835fa480e0c2ddefb040ee82d58b3 2015-09-16T12:56:42Z -github.com/juju/loggo git 8232ab8918d91c72af1a9fb94d3edbe31d88b790 2017-06-05T01:46:07Z -github.com/juju/retry git 62c62032529169c7ec02fa48f93349604c345e1f 2015-10-29T02:48:21Z -github.com/juju/schema git 075de04f9b7d7580d60a1e12a0b3f50bb18e6998 2016-04-20T04:42:03Z -github.com/juju/testing git 44801989f0f7f280bd16b58e898ba9337807f147 2018-04-02T13:06:37Z -github.com/juju/utils git 2000ea4ff0431598aec2b7e1d11d5d49b5384d63 2018-04-24T09:41:59Z -github.com/juju/version git 1f41e27e54f21acccf9b2dddae063a782a8a7ceb 2016-10-31T05:19:06Z -golang.org/x/crypto git 650f4a345ab4e5b245a3034b110ebc7299e68186 2018-02-14T00:00:28Z -golang.org/x/net git 61147c48b25b599e5b561d2e9c4f3e1ef489ca41 2018-04-06T21:48:16Z -gopkg.in/check.v1 git 4f90aeace3a26ad7021961c297b22c42160c7b25 2016-01-05T16:49:36Z -gopkg.in/mgo.v2 git f2b6f6c918c452ad107eec89615f074e3bd80e33 2016-08-18T01:52:18Z -gopkg.in/yaml.v2 git 1be3d31502d6eabc0dd7ce5b0daab022e14a5538 2017-07-12T05:45:46Z diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..db3b6b3 --- /dev/null +++ b/go.mod @@ -0,0 +1,19 @@ +module github.com/seanhoughton/gomaasapi + +go 1.13 + +require ( + github.com/juju/collections v0.0.0-20180515203731-520e0549d51a + github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 + github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 + github.com/juju/retry v0.0.0-20151029024821-62c620325291 + github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d + github.com/juju/testing v0.0.0-20180402130637-44801989f0f7 + github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043 + github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2 + golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4 + golang.org/x/net v0.0.0-20180406214816-61147c48b25b + gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2 + gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 + gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..6adaa8c --- /dev/null +++ b/go.sum @@ -0,0 +1,19 @@ +github.com/juju/collections v0.0.0-20180515203731-520e0549d51a h1:PPCCWrZzJMhFu4PxX3vRM65dq7LZMyreWAMPsvttUQk= +github.com/juju/collections v0.0.0-20180515203731-520e0549d51a/go.mod h1:Ep+c0vnxsgmmTtsMibPgEEleZyi0b4uVvyzJ+8ka9EI= +github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 h1:Sem5Flzxj8ZdAgY2wfHBUlOYyP4wrpIfM8IZgANNGh8= +github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18/go.mod h1:W54LbzXuIE0boCoNJfwqpmkKJ1O4TCTZMetAt6jGk7Q= +github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 h1:Y+lzErDTURqeXqlqYi4YBYbDd7ycU74gW1ADt57/bgY= +github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U= +github.com/juju/retry v0.0.0-20151029024821-62c620325291/go.mod h1:OohPQGsr4pnxwD5YljhQ+TZnuVRYpa5irjugL1Yuif4= +github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d h1:JYANSZLNBXFgnNfGDOUAV+atWFDmOqJ1WPNmyS+YCCw= +github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d/go.mod h1:7dL+43wADDfx5rD9ibr5H9Dgr4iOM3uHOa1i4IVLak8= +github.com/juju/testing v0.0.0-20180402130637-44801989f0f7/go.mod h1:63prj8cnj0tU0S9OHjGJn+b1h0ZghCndfnbQolrYTwA= +github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043/go.mod h1:6/KLg8Wz/y2KVGWEpkK9vMNGkOnu4k/cqs8Z1fKjTOk= +github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2 h1:loQDi5MyxxNm7Q42mBGuPD6X+F6zw8j5S9yexLgn/BE= +github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2/go.mod h1:kE8gK5X0CImdr7qpSKl3xB2PmpySSmfj7zVbkZFs81U= +golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/net v0.0.0-20180406214816-61147c48b25b/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 h1:hILp2hNrRnYjZpmIbx70psAHbBSEcQ1NIzDcUbJ1b6g= +gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4/go.mod h1:yeKp02qBN3iKW1OzL3MGk2IdtZzaj7SFntXj72NppTA= +gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= From b941989720111fe104c9d1d2059f06b4b19c4c0b Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 24 Jan 2020 14:06:16 -0800 Subject: [PATCH 03/21] Add machine.CreateBond --- interface.go | 38 +++++++++++++++++++++++++++++++------- interfaces.go | 4 ++++ machine.go | 49 +++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 84 insertions(+), 7 deletions(-) diff --git a/interface.go b/interface.go index 9865833..e59f010 100644 --- a/interface.go +++ b/interface.go @@ -113,9 +113,17 @@ func (i *interface_) EffectiveMTU() int { // UpdateInterfaceArgs is an argument struct for calling Interface.Update. type UpdateInterfaceArgs struct { - Name string - MACAddress string - VLAN VLAN + Name string + MACAddress string + VLAN VLAN + BridgeSTP bool + BridgeFD int + BondMiimon int + BondDownDelay int + BondUpDelay int + BondLACPRate string + BondXmitHashPolicy string + BondMode string } func (a *UpdateInterfaceArgs) vlanID() int { @@ -125,16 +133,32 @@ func (a *UpdateInterfaceArgs) vlanID() int { return a.VLAN.ID() } +func (a *UpdateInterfaceArgs) toParams() *URLParams { + params := NewURLParams() + params.MaybeAdd("name", a.Name) + params.MaybeAdd("mac_address", a.MACAddress) + params.MaybeAddInt("vlan", a.vlanID()) + if a.BridgeSTP { + params.MaybeAdd("bridge_stp", "1") + } + params.MaybeAddInt("bridge_fd", a.BridgeFD) + params.MaybeAddInt("bond_miimon ", a.BondMiimon) + params.MaybeAddInt("bond_down_delay", a.BondDownDelay) + params.MaybeAddInt("bond_up_delay", a.BondUpDelay) + params.MaybeAdd("bond_lacp_rate", a.BondLACPRate) + params.MaybeAdd("bond_xmit_hash_policy", a.BondXmitHashPolicy) + params.MaybeAdd("bond_mode", a.BondMode) + return params +} + // Update implements Interface. func (i *interface_) Update(args UpdateInterfaceArgs) error { var empty UpdateInterfaceArgs if args == empty { return nil } - params := NewURLParams() - params.MaybeAdd("name", args.Name) - params.MaybeAdd("mac_address", args.MACAddress) - params.MaybeAddInt("vlan", args.vlanID()) + + params := args.toParams() source, err := i.controller.put(i.resourceURI, params.Values) if err != nil { if svrErr, ok := errors.Cause(err).(ServerError); ok { diff --git a/interfaces.go b/interfaces.go index ee3a225..886127c 100644 --- a/interfaces.go +++ b/interfaces.go @@ -234,6 +234,10 @@ type Machine interface { // specified. If there is no match, nil is returned. Interface(id int) Interface + // CreateBond creates a bond with the provided interfaces and returns the + // newly created bond interface. + CreateBond(args CreateMachineBondArgs) (Interface, error) + // PhysicalBlockDevices returns all the physical block devices on the machine. PhysicalBlockDevices() []BlockDevice // PhysicalBlockDevice returns the physical block device for the machine diff --git a/machine.go b/machine.go index c2c5bdb..0801709 100644 --- a/machine.go +++ b/machine.go @@ -7,6 +7,7 @@ import ( "fmt" "net/http" "net/url" + "strings" "github.com/juju/errors" "github.com/juju/schema" @@ -287,6 +288,54 @@ func (m *machine) Start(args StartArgs) error { return nil } +// CreateMachineBondArgs is the argument structure for Machine.CreateBond +type CreateMachineBondArgs struct { + UpdateInterfaceArgs + Parents []Interface +} + +func (a *CreateMachineBondArgs) toParams() *URLParams { + params := a.UpdateInterfaceArgs.toParams() + parents := []string{} + for _, p := range a.Parents { + parents = append(parents, fmt.Sprintf("%d", p.ID())) + } + params.MaybeAdd("parents", strings.Join(parents, ",")) + return params +} + +// Validate ensures that all required values are non-emtpy. +func (a *CreateMachineBondArgs) Validate() error { + return nil +} + +// CreateBond implements Machine +func (m *machine) CreateBond(args CreateMachineBondArgs) (_ Interface, err error) { + if err := args.Validate(); err != nil { + return nil, errors.Trace(err) + } + + params := args.toParams() + source, err := m.controller.post(m.resourceURI+"interfaces/", "create_bond", params.Values) + 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) + } + + response, err := readInterface(m.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + return response, nil +} + // CreateMachineDeviceArgs is an argument structure for Machine.CreateDevice. // Only InterfaceName and MACAddress fields are required, the others are only // used if set. If Subnet and VLAN are both set, Subnet.VLAN() must match the From 20403553ce969ef6f25613b71bc61534428e876e Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 24 Jan 2020 14:06:30 -0800 Subject: [PATCH 04/21] Add machine.Commission --- interfaces.go | 3 +++ machine.go | 42 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 45 insertions(+) diff --git a/interfaces.go b/interfaces.go index 886127c..d55fd7a 100644 --- a/interfaces.go +++ b/interfaces.go @@ -257,6 +257,9 @@ type Machine interface { Zone() Zone Pool() Pool + // Commision makes a new node Ready + Commission(CommissionArgs) error + // Start the machine and install the operating system specified in the args. Start(StartArgs) error diff --git a/machine.go b/machine.go index 0801709..3ce6736 100644 --- a/machine.go +++ b/machine.go @@ -248,6 +248,48 @@ func (m *machine) Devices(args DevicesArgs) ([]Device, error) { return result, nil } +// CommisionArgs is an argument struct for Machine.Commission +type CommissionArgs struct { + EnableSSH bool + SkipBMCConfig bool + SkipNetworking bool + SkipStorage bool + CommissioningScripts []string + TestingScripts []string +} + +func (m *machine) Commission(args CommissionArgs) error { + params := NewURLParams() + params.MaybeAddBool("enableSSH", args.EnableSSH) + params.MaybeAddBool("skip_bmc_config", args.SkipBMCConfig) + params.MaybeAddBool("skip_networking", args.SkipNetworking) + params.MaybeAddBool("skip_storage", args.SkipStorage) + params.MaybeAdd("commissioning_scripts", strings.Join(args.CommissioningScripts, ",")) + params.MaybeAdd("testing_scripts", strings.Join(args.TestingScripts, ",")) + + result, err := m.controller.post(m.resourceURI, "commission", params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound, http.StatusConflict: + return errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return NewUnexpectedError(err) + } + + machine, err := readMachine(m.controller.apiVersion, result) + if err != nil { + return errors.Trace(err) + } + m.updateFrom(machine) + return nil +} + // StartArgs is an argument struct for passing parameters to the Machine.Start // method. type StartArgs struct { From 6819c3ddebbc53bc8e1e13fbeffd3cb29d1640d8 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 24 Jan 2020 14:24:18 -0800 Subject: [PATCH 05/21] Add machine.Update --- interfaces.go | 3 +++ machine.go | 50 +++++++++++++++++++++++++++++++++++++++++++++++++- 2 files changed, 52 insertions(+), 1 deletion(-) diff --git a/interfaces.go b/interfaces.go index d55fd7a..6c915e9 100644 --- a/interfaces.go +++ b/interfaces.go @@ -234,6 +234,9 @@ type Machine interface { // specified. If there is no match, nil is returned. Interface(id int) Interface + // Update allows editing of some of the machine's properties + Update(args UpdateMachineArgs) error + // CreateBond creates a bond with the provided interfaces and returns the // newly created bond interface. CreateBond(args CreateMachineBondArgs) (Interface, error) diff --git a/machine.go b/machine.go index 3ce6736..7a54d2a 100644 --- a/machine.go +++ b/machine.go @@ -248,7 +248,55 @@ func (m *machine) Devices(args DevicesArgs) ([]Device, error) { return result, nil } -// CommisionArgs is an argument struct for Machine.Commission +// UpdateMachineArgs is arguments for machine.Update +type UpdateMachineArgs struct { + Hostname string + Domain string + PowerType string + PowerAddress string + PowerUser string + PowerPassword string + PowerOpts map[string]string +} + +// Update implementes Machine +func (m *machine) Update(args UpdateMachineArgs) error { + params := NewURLParams() + params.MaybeAdd("hostname", args.Hostname) + params.MaybeAdd("domain", args.Domain) + params.MaybeAdd("power_type", args.PowerType) + params.MaybeAdd("power_parameters_power_user", args.PowerUser) + params.MaybeAdd("power_parameters_power_password", args.PowerUser) + params.MaybeAdd("power_parameters_power_address", args.PowerAddress) + if args.PowerOpts != nil { + for k, v := range args.PowerOpts { + params.MaybeAdd(fmt.Sprintf("power_parameters_%s", k), v) + } + } + result, err := m.controller.put(m.resourceURI, params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound, http.StatusConflict: + return errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return NewUnexpectedError(err) + } + + machine, err := readMachine(m.controller.apiVersion, result) + if err != nil { + return errors.Trace(err) + } + m.updateFrom(machine) + return nil +} + +// CommissionArgs is an argument struct for Machine.Commission type CommissionArgs struct { EnableSSH bool SkipBMCConfig bool From 37ccf2b10d17d53dfb5a84d55bfc3272ad425908 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 28 Jan 2020 08:09:46 -0800 Subject: [PATCH 06/21] Add tags api --- controller.go | 60 ++++++++++++++++++++ interfaces.go | 19 +++++++ tag.go | 150 ++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 229 insertions(+) create mode 100644 tag.go diff --git a/controller.go b/controller.go index 912e4c4..8651659 100644 --- a/controller.go +++ b/controller.go @@ -754,6 +754,66 @@ func (c *controller) AddFile(args AddFileArgs) error { return nil } +func (c *controller) Tags() ([]Tag, error) { + source, err := c.getQuery("tags", url.Values{}) + if err != nil { + return nil, NewUnexpectedError(err) + } + tags, err := readTags(c.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + var result []Tag + for _, t := range tags { + t.controller = c + result = append(result, t) + } + return result, nil +} + +func (c *controller) GetTag(name string) (Tag, error) { + source, err := c.getQuery("tags/"+name, url.Values{}) + if err != nil { + return nil, NewUnexpectedError(err) + } + tag, err := readTag(c.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + tag.controller = c + return tag, nil +} + +// CreateTagArgs are creation parameters +type CreateTagArgs struct { + Name string + Comment string + Definition string +} + +// Validate ensures arguments are valid +func (a *CreateTagArgs) Validate() error { + if a.Name == "" { + return fmt.Errorf("Missing name value") + } + return nil +} + +func (c *controller) CreateTag(args CreateTagArgs) (Tag, error) { + if err := args.Validate(); err != nil { + return nil, err + } + params := NewURLParams() + params.MaybeAdd("name", args.Name) + params.MaybeAdd("comment", args.Comment) + params.MaybeAdd("definition", args.Definition) + result, err := c.post("tags", "", params.Values) + if err != nil { + return nil, err + } + return readTag(c.apiVersion, result) +} + func (c *controller) checkCreds() error { if _, err := c.getOp("users", "whoami"); err != nil { if svrErr, ok := errors.Cause(err).(ServerError); ok { diff --git a/interfaces.go b/interfaces.go index 6c915e9..f18630a 100644 --- a/interfaces.go +++ b/interfaces.go @@ -72,6 +72,15 @@ type Controller interface { // Returns the DNS Domain Managed By MAAS Domains() ([]Domain, error) + + // Returns the set of all tags + Tags() ([]Tag, error) + + // Retuns a aspecific tag or an error if it doesn't exist + GetTag(name string) (Tag, error) + + // Creates a new tag, or returns an error if the tag already exists + CreateTag(args CreateTagArgs) (Tag, error) } // File represents a file stored in the MAAS controller. @@ -427,3 +436,13 @@ type OwnerDataHolder interface { // released. SetOwnerData(map[string]string) error } + +// Tag represents a MaaS device tag +type Tag interface { + Name() string + Definition() string + Comment() string + Machines() ([]Machine, error) + AddToMachine(Machine) error + RemoveFromMachine(Machine) error +} diff --git a/tag.go b/tag.go new file mode 100644 index 0000000..e821717 --- /dev/null +++ b/tag.go @@ -0,0 +1,150 @@ +package gomaasapi + +import ( + "net/url" + + "github.com/juju/errors" + "github.com/juju/schema" + "github.com/juju/version" +) + +type tag struct { + controller *controller + resourceURI string + + name string + definition string + comment string +} + +// Name implements Tag. +func (s *tag) Name() string { + return s.name +} + +// Definition implements Tag. +func (s *tag) Definition() string { + return s.definition +} + +// Comment implements Tag. +func (s *tag) Comment() string { + return s.definition +} + +func (s *tag) Machines() ([]Machine, error) { + source, err := s.controller.getOp(s.resourceURI, "machines") + if err != nil { + return nil, NewUnexpectedError(err) + } + machines, err := readMachines(s.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + var result []Machine + for _, m := range machines { + //m.controller = c + result = append(result, m) + } + return result, nil +} + +func (s *tag) AddToMachine(machine Machine) error { + params := url.Values{ + "add": []string{machine.SystemID()}, + } + _, err := s.controller.post(s.resourceURI, "update_nodes", params) + return err +} + +func (s *tag) RemoveFromMachine(machine Machine) error { + params := url.Values{ + "remove": []string{machine.SystemID()}, + } + _, err := s.controller.post(s.resourceURI, "update_nodes", params) + return err +} + +func readTags(controllerVersion version.Number, source interface{}) ([]*tag, error) { + checker := schema.List(schema.StringMap(schema.Any())) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, errors.Annotatef(err, "tag base schema check failed") + } + valid := coerced.([]interface{}) + + var deserialisationVersion version.Number + for v := range tagDeserializationFuncs { + if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 { + deserialisationVersion = v + } + } + if deserialisationVersion == version.Zero { + return nil, errors.Errorf("no tag read func for version %s", controllerVersion) + } + readFunc := tagDeserializationFuncs[deserialisationVersion] + return readTagList(valid, readFunc) +} + +func readTag(controllerVersion version.Number, source interface{}) (*tag, error) { + var deserialisationVersion version.Number + for v := range tagDeserializationFuncs { + if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 { + deserialisationVersion = v + } + } + + if deserialisationVersion == version.Zero { + return nil, errors.Errorf("no tag read func for version %s", controllerVersion) + } + readFunc := tagDeserializationFuncs[deserialisationVersion] + return readFunc(source.(map[string]interface{})) +} + +// readTagList expects the values of the sourceList to be string maps. +func readTagList(sourceList []interface{}, readFunc tagDeserializationFunc) ([]*tag, error) { + result := make([]*tag, 0, len(sourceList)) + for i, value := range sourceList { + source, ok := value.(map[string]interface{}) + if !ok { + return nil, errors.Errorf("unexpected value for tag %d, %T", i, value) + } + tag, err := readFunc(source) + if err != nil { + return nil, errors.Annotatef(err, "tag %d", i) + } + result = append(result, tag) + } + return result, nil +} + +type tagDeserializationFunc func(map[string]interface{}) (*tag, error) + +var tagDeserializationFuncs = map[version.Number]tagDeserializationFunc{ + twoDotOh: tag_2_0, +} + +func tag_2_0(source map[string]interface{}) (*tag, error) { + fields := schema.Fields{ + "resource_uri": schema.String(), + "name": schema.String(), + "definition": schema.String(), + "comment": schema.String(), + } + checker := schema.FieldMap(fields, nil) // no defaults + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, errors.Annotatef(err, "tag 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. + + result := &tag{ + resourceURI: valid["resource_uri"].(string), + name: valid["name"].(string), + comment: valid["comment"].(string), + definition: valid["definition"].(string), + } + return result, nil +} From c87b2dbab66190669e553572ea63490c5e2f92b1 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 28 Jan 2020 14:34:32 -0800 Subject: [PATCH 07/21] Add machine creation --- controller.go | 55 +++++++++++++++++++++++++++++++++++++++++++++++++++ interfaces.go | 3 +++ machine.go | 31 +++++++++++++++++++---------- 3 files changed, 79 insertions(+), 10 deletions(-) diff --git a/controller.go b/controller.go index 8651659..5929fed 100644 --- a/controller.go +++ b/controller.go @@ -335,6 +335,61 @@ func (c *controller) CreateDevice(args CreateDeviceArgs) (Device, error) { return device, nil } +// CreateMachineArgs is a argument struct for passing information into CreateDevice. +type CreateMachineArgs struct { + UpdateMachineArgs + Architecture string + Description string + MACAddresses []string +} + +// Validate ensures the arguments are acceptable +func (a *CreateMachineArgs) Validate() error { + if err := a.UpdateMachineArgs.Validate(); err != nil { + return err + } + if len(a.MACAddresses) == 0 { + return fmt.Errorf("at least one MAC address must be specified") + } + + return nil +} + +// ToParams converts arguments to URL parameters +func (a *CreateMachineArgs) ToParams() *URLParams { + params := a.UpdateMachineArgs.ToParams() + params.MaybeAdd("architecture", a.Architecture) + params.MaybeAdd("description", a.Description) + params.MaybeAddMany("mac_addresses", a.MACAddresses) + return params +} + +// CreateMachine implements Controller. +func (c *controller) CreateMachine(args CreateMachineArgs) (Machine, error) { + // There must be at least one mac address. + if err := args.Validate(); err != nil { + return nil, errors.NewBadRequest(err, "Invalid CreateMachine arguments") + } + params := args.ToParams() + result, err := c.post("machines", "", params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + if svrErr.StatusCode == http.StatusBadRequest { + return nil, errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + } + } + // Translate http errors. + return nil, NewUnexpectedError(err) + } + + machine, err := readMachine(c.apiVersion, result) + if err != nil { + return nil, errors.Trace(err) + } + machine.controller = c + return machine, nil +} + // MachinesArgs is a argument struct for selecting Machines. // Only machines that match the specified criteria are returned. type MachinesArgs struct { diff --git a/interfaces.go b/interfaces.go index f18630a..3fbc808 100644 --- a/interfaces.go +++ b/interfaces.go @@ -52,6 +52,9 @@ type Controller interface { // from the user making them available to be allocated again. ReleaseMachines(ReleaseMachinesArgs) error + // CreateMachine will create a new machine with the provided parameters + CreateMachine(CreateMachineArgs) (Machine, error) + // Devices returns a list of devices that match the params. Devices(DevicesArgs) ([]Device, error) diff --git a/machine.go b/machine.go index 7a54d2a..eaac0fc 100644 --- a/machine.go +++ b/machine.go @@ -259,20 +259,31 @@ type UpdateMachineArgs struct { PowerOpts map[string]string } -// Update implementes Machine -func (m *machine) Update(args UpdateMachineArgs) error { +// Validate ensures the arguments are acceptable +func (a *UpdateMachineArgs) Validate() error { + return nil +} + +// ToParams converts arguments to URL parameters +func (a *UpdateMachineArgs) ToParams() *URLParams { params := NewURLParams() - params.MaybeAdd("hostname", args.Hostname) - params.MaybeAdd("domain", args.Domain) - params.MaybeAdd("power_type", args.PowerType) - params.MaybeAdd("power_parameters_power_user", args.PowerUser) - params.MaybeAdd("power_parameters_power_password", args.PowerUser) - params.MaybeAdd("power_parameters_power_address", args.PowerAddress) - if args.PowerOpts != nil { - for k, v := range args.PowerOpts { + params.MaybeAdd("hostname", a.Hostname) + params.MaybeAdd("domain", a.Domain) + params.MaybeAdd("power_type", a.PowerType) + params.MaybeAdd("power_parameters_power_user", a.PowerUser) + params.MaybeAdd("power_parameters_power_password", a.PowerUser) + params.MaybeAdd("power_parameters_power_address", a.PowerAddress) + if a.PowerOpts != nil { + for k, v := range a.PowerOpts { params.MaybeAdd(fmt.Sprintf("power_parameters_%s", k), v) } } + return params +} + +// Update implementes Machine +func (m *machine) Update(args UpdateMachineArgs) error { + params := args.ToParams() result, err := m.controller.put(m.resourceURI, params.Values) if err != nil { if svrErr, ok := errors.Cause(err).(ServerError); ok { From dc8a46991fff335661de96088221c5057bf28480 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 28 Jan 2020 15:09:27 -0800 Subject: [PATCH 08/21] Add erase options to the ReleaseMachines function --- controller.go | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/controller.go b/controller.go index 5929fed..06fe85f 100644 --- a/controller.go +++ b/controller.go @@ -672,8 +672,12 @@ func (c *controller) AllocateMachine(args AllocateMachineArgs) (Machine, Constra // ReleaseMachinesArgs is an argument struct for passing the machine system IDs // and an optional comment into the ReleaseMachines method. type ReleaseMachinesArgs struct { - SystemIDs []string - Comment string + SystemIDs []string + Comment string + Erase bool + SecureErase bool + QuickErase bool + Force bool } // ReleaseMachines implements Controller. @@ -686,6 +690,10 @@ func (c *controller) ReleaseMachines(args ReleaseMachinesArgs) error { params := NewURLParams() params.MaybeAddMany("machines", args.SystemIDs) params.MaybeAdd("comment", args.Comment) + params.MaybeAddBool("erase", args.Erase) + params.MaybeAddBool("secure_erase", args.SecureErase) + params.MaybeAddBool("quick_erase", args.QuickErase) + params.MaybeAddBool("force", args.Force) _, err := c.post("machines", "release", params.Values) if err != nil { if svrErr, ok := errors.Cause(err).(ServerError); ok { From af20fc472d10a6ef2ef868a545092a4b6d1a70ca Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 28 Jan 2020 15:25:08 -0800 Subject: [PATCH 09/21] Add machine.Delete --- interfaces.go | 7 +++++-- machine.go | 8 ++++++++ tag.go | 8 ++++---- 3 files changed, 17 insertions(+), 6 deletions(-) diff --git a/interfaces.go b/interfaces.go index 3fbc808..0de7397 100644 --- a/interfaces.go +++ b/interfaces.go @@ -281,6 +281,9 @@ type Machine interface { // CreateDevice creates a new Device with this Machine as the parent. // The device will have one interface that is linked to the specified subnet. CreateDevice(CreateMachineDeviceArgs) (Device, error) + + // Delete removes the machine from maas + Delete() error } // Space is a name for a collection of Subnets. @@ -446,6 +449,6 @@ type Tag interface { Definition() string Comment() string Machines() ([]Machine, error) - AddToMachine(Machine) error - RemoveFromMachine(Machine) error + AddToMachine(systemID string) error + RemoveFromMachine(systemID string) error } diff --git a/machine.go b/machine.go index eaac0fc..de35912 100644 --- a/machine.go +++ b/machine.go @@ -581,6 +581,14 @@ func (m *machine) SetOwnerData(ownerData map[string]string) error { return nil } +func (m *machine) Delete() error { + err := m.controller.delete(m.resourceURI) + if err != nil { + return errors.Trace(err) + } + return nil +} + func readMachine(controllerVersion version.Number, source interface{}) (*machine, error) { readFunc, err := getMachineDeserializationFunc(controllerVersion) if err != nil { diff --git a/tag.go b/tag.go index e821717..385a6bc 100644 --- a/tag.go +++ b/tag.go @@ -49,17 +49,17 @@ func (s *tag) Machines() ([]Machine, error) { return result, nil } -func (s *tag) AddToMachine(machine Machine) error { +func (s *tag) AddToMachine(systemID string) error { params := url.Values{ - "add": []string{machine.SystemID()}, + "add": []string{systemID}, } _, err := s.controller.post(s.resourceURI, "update_nodes", params) return err } -func (s *tag) RemoveFromMachine(machine Machine) error { +func (s *tag) RemoveFromMachine(systemID string) error { params := url.Values{ - "remove": []string{machine.SystemID()}, + "remove": []string{systemID}, } _, err := s.controller.post(s.resourceURI, "update_nodes", params) return err From 1654585f64e449ef0a7f0e3a7700e07d7fe61966 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Wed, 29 Jan 2020 16:06:01 -0800 Subject: [PATCH 10/21] Add more CreateMachine arguments --- controller.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/controller.go b/controller.go index 06fe85f..74739b9 100644 --- a/controller.go +++ b/controller.go @@ -340,6 +340,7 @@ type CreateMachineArgs struct { UpdateMachineArgs Architecture string Description string + Commission bool MACAddresses []string } @@ -361,6 +362,11 @@ func (a *CreateMachineArgs) ToParams() *URLParams { params.MaybeAdd("architecture", a.Architecture) params.MaybeAdd("description", a.Description) params.MaybeAddMany("mac_addresses", a.MACAddresses) + if a.Commission { + params.MaybeAdd("commission", "true") + } else { + params.MaybeAdd("commission", "false") + } return params } From e41d3af37923def36e4eeb074fa1ba544cca72ac Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 31 Jan 2020 14:24:59 -0800 Subject: [PATCH 11/21] Add controller.GetMachine --- controller.go | 13 +++++++++++++ interfaces.go | 3 +++ 2 files changed, 16 insertions(+) diff --git a/controller.go b/controller.go index 74739b9..d3d9e63 100644 --- a/controller.go +++ b/controller.go @@ -439,6 +439,19 @@ func (c *controller) Machines(args MachinesArgs) ([]Machine, error) { return result, nil } +func (c *controller) GetMachine(systemID string) (Machine, error) { + source, err := c.getQuery("machines/"+systemID, url.Values{}) + if err != nil { + return nil, NewUnexpectedError(err) + } + m, err := readMachine(c.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + m.controller = c + return m, nil +} + func ownerDataMatches(ownerData, filter map[string]string) bool { for key, value := range filter { if ownerData[key] != value { diff --git a/interfaces.go b/interfaces.go index 0de7397..38d26da 100644 --- a/interfaces.go +++ b/interfaces.go @@ -44,6 +44,9 @@ type Controller interface { // Machines returns a list of machines that match the params. Machines(MachinesArgs) ([]Machine, error) + // GetMachine gets a single machine + GetMachine(systemID string) (Machine, error) + // AllocateMachine will attempt to allocate a machine to the user. // If successful, the allocated machine is returned. AllocateMachine(AllocateMachineArgs) (Machine, ConstraintMatches, error) From faf88ac35a2e8d76814f5d8dc8a772527e726fdb Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Mon, 3 Feb 2020 14:33:56 -0800 Subject: [PATCH 12/21] Add machine.HWEKernel accessor --- interfaces.go | 1 + machine.go | 9 +++++++++ 2 files changed, 10 insertions(+) diff --git a/interfaces.go b/interfaces.go index 38d26da..b42420b 100644 --- a/interfaces.go +++ b/interfaces.go @@ -223,6 +223,7 @@ type Machine interface { OperatingSystem() string DistroSeries() string + HWEKernel() string Architecture() string Memory() int CPUCount() int diff --git a/machine.go b/machine.go index de35912..11d922a 100644 --- a/machine.go +++ b/machine.go @@ -27,6 +27,7 @@ type machine struct { operatingSystem string distroSeries string + hweKernel string architecture string memory int cpuCount int @@ -163,6 +164,11 @@ func (m *machine) DistroSeries() string { return m.distroSeries } +// HWEKernel implements Machine. +func (m *machine) HWEKernel() string { + return m.hweKernel +} + // Architecture implements Machine. func (m *machine) Architecture() string { return m.architecture @@ -666,6 +672,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { "osystem": schema.String(), "distro_series": schema.String(), + "hwe_kernel": schema.OneOf(schema.Nil(""), schema.String()), "architecture": schema.OneOf(schema.Nil(""), schema.String()), "memory": schema.ForceInt(), "cpu_count": schema.ForceInt(), @@ -732,6 +739,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { } architecture, _ := valid["architecture"].(string) statusMessage, _ := valid["status_message"].(string) + hweKernel, _ := valid["hwe_kernel"].(string) result := &machine{ resourceURI: valid["resource_uri"].(string), @@ -743,6 +751,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { operatingSystem: valid["osystem"].(string), distroSeries: valid["distro_series"].(string), + hweKernel: hweKernel, architecture: architecture, memory: valid["memory"].(int), cpuCount: valid["cpu_count"].(int), From b9a02ff36972c0b9d0ea0223ca2691a8879f36f0 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Fri, 7 Feb 2020 09:55:25 -0800 Subject: [PATCH 13/21] Use retryablehttp to manage retries This handles more cases, including 504 and other timeouts --- client.go | 84 +++++++++++++++++-------------------------------------- go.mod | 12 ++++---- go.sum | 18 ++++++++++++ oauth.go | 6 ++-- 4 files changed, 54 insertions(+), 66 deletions(-) diff --git a/client.go b/client.go index eab8d87..4d7855d 100644 --- a/client.go +++ b/client.go @@ -12,27 +12,18 @@ import ( "net/http" "net/url" "regexp" - "strconv" "strings" - "time" + "github.com/hashicorp/go-retryablehttp" "github.com/juju/errors" ) -const ( - // Number of retries performed when the server returns a 503 - // response with a 'Retry-after' header. A request will be issued - // at most NumberOfRetries + 1 times. - NumberOfRetries = 4 - - RetryAfterHeaderName = "Retry-After" -) - // Client represents a way to communicating with a MAAS API instance. // It is stateless, so it can have concurrent requests in progress. type Client struct { - APIURL *url.URL - Signer OAuthSigner + APIURL *url.URL + Signer OAuthSigner + httpClient *retryablehttp.Client } // ServerError is an http error (or at least, a non-2xx result) received from @@ -70,48 +61,14 @@ func readAndClose(stream io.ReadCloser) ([]byte, error) { // returned error will be ServerError and the returned body will reflect the // server's response. If the server returns a 503 response with a 'Retry-after' // header, the request will be transparenty retried. -func (client Client) dispatchRequest(request *http.Request) ([]byte, error) { - // First, store the request's body into a byte[] to be able to restore it - // after each request. - bodyContent, err := readAndClose(request.Body) - if err != nil { - return nil, err - } - for retry := 0; retry < NumberOfRetries; retry++ { - // Restore body before issuing request. - newBody := ioutil.NopCloser(bytes.NewReader(bodyContent)) - request.Body = newBody - body, err := client.dispatchSingleRequest(request) - // If this is a 503 response with a non-void "Retry-After" header: wait - // as instructed and retry the request. - if err != nil { - serverError, ok := errors.Cause(err).(ServerError) - if ok && serverError.StatusCode == http.StatusServiceUnavailable { - retry_time_int, errConv := strconv.Atoi(serverError.Header.Get(RetryAfterHeaderName)) - if errConv == nil { - select { - case <-time.After(time.Duration(retry_time_int) * time.Second): - } - continue - } - } - } - return body, err - } - // Restore body before issuing request. - newBody := ioutil.NopCloser(bytes.NewReader(bodyContent)) - request.Body = newBody - return client.dispatchSingleRequest(request) -} +func (client Client) dispatchRequest(request *retryablehttp.Request) ([]byte, error) { + client.Signer.OAuthSign(&request.Header) -func (client Client) dispatchSingleRequest(request *http.Request) ([]byte, error) { - client.Signer.OAuthSign(request) - httpClient := http.Client{} // See https://code.google.com/p/go/issues/detail?id=4677 // We need to force the connection to close each time so that we don't // hit the above Go bug. request.Close = true - response, err := httpClient.Do(request) + response, err := client.httpClient.Do(request) if err != nil { return nil, err } @@ -148,9 +105,9 @@ func (client Client) Get(uri *url.URL, operation string, parameters url.Values) if operation != "" { parameters.Set("op", operation) } - queryUrl := client.GetURL(uri) - queryUrl.RawQuery = parameters.Encode() - request, err := http.NewRequest("GET", queryUrl.String(), nil) + queryURL := client.GetURL(uri) + queryURL.RawQuery = parameters.Encode() + request, err := retryablehttp.NewRequest("GET", queryURL.String(), nil) if err != nil { return nil, err } @@ -204,7 +161,7 @@ func (client Client) nonIdempotentRequestFiles(method string, uri *url.URL, para } writer.Close() url := client.GetURL(uri) - request, err := http.NewRequest(method, url.String(), buf) + request, err := retryablehttp.NewRequest(method, url.String(), buf) if err != nil { return nil, err } @@ -217,7 +174,7 @@ func (client Client) nonIdempotentRequestFiles(method string, uri *url.URL, para // requests (but not GET or DELETE requests). func (client Client) nonIdempotentRequest(method string, uri *url.URL, parameters url.Values) ([]byte, error) { url := client.GetURL(uri) - request, err := http.NewRequest(method, url.String(), strings.NewReader(string(parameters.Encode()))) + request, err := retryablehttp.NewRequest(method, url.String(), strings.NewReader(string(parameters.Encode()))) if err != nil { return nil, err } @@ -245,7 +202,7 @@ func (client Client) Put(uri *url.URL, parameters url.Values) ([]byte, error) { // Delete deletes an object on the API, using an HTTP "DELETE" request. func (client Client) Delete(uri *url.URL) error { url := client.GetURL(uri) - request, err := http.NewRequest("DELETE", url.String(), strings.NewReader("")) + request, err := retryablehttp.NewRequest("DELETE", url.String(), strings.NewReader("")) if err != nil { return err } @@ -259,7 +216,7 @@ func (client Client) Delete(uri *url.URL) error { // Anonymous "signature method" implementation. type anonSigner struct{} -func (signer anonSigner) OAuthSign(request *http.Request) error { +func (signer anonSigner) OAuthSign(request *http.Header) error { return nil } @@ -330,5 +287,16 @@ func NewAuthenticatedClient(versionedURL, apiKey string) (*Client, error) { if err != nil { return nil, err } - return &Client{Signer: signer, APIURL: parsedURL}, nil + + httpClient := retryablehttp.NewClient() + + // Need to re-sign the request before each retry + httpClient.RequestLogHook = func(logger retryablehttp.Logger, request *http.Request, count int) { + err := signer.OAuthSign(&request.Header) + if err != nil { + logger.Printf("[ERROR] Failed to sign request: %v", err) + } + } + + return &Client{Signer: signer, APIURL: parsedURL, httpClient: httpClient}, nil } diff --git a/go.mod b/go.mod index db3b6b3..eee3595 100644 --- a/go.mod +++ b/go.mod @@ -3,17 +3,19 @@ module github.com/seanhoughton/gomaasapi go 1.13 require ( + github.com/hashicorp/go-retryablehttp v0.6.4 github.com/juju/collections v0.0.0-20180515203731-520e0549d51a github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 + github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 - github.com/juju/retry v0.0.0-20151029024821-62c620325291 + github.com/juju/retry v0.0.0-20151029024821-62c620325291 // indirect github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d github.com/juju/testing v0.0.0-20180402130637-44801989f0f7 - github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043 + github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043 // indirect github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2 - golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4 - golang.org/x/net v0.0.0-20180406214816-61147c48b25b + golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4 // indirect + golang.org/x/net v0.0.0-20180406214816-61147c48b25b // indirect gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2 gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 - gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 + gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 // indirect ) diff --git a/go.sum b/go.sum index 6adaa8c..1985ad0 100644 --- a/go.sum +++ b/go.sum @@ -1,19 +1,37 @@ +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/hashicorp/go-cleanhttp v0.5.1 h1:dH3aiDG9Jvb5r5+bYHsikaOUIpcM0xvgMXVoDkXMzJM= +github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-hclog v0.9.2 h1:CG6TE5H9/JXsFWJCfoIVpKFIkFe6ysEuHirp4DxCsHI= +github.com/hashicorp/go-hclog v0.9.2/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= +github.com/hashicorp/go-retryablehttp v0.6.4 h1:BbgctKO892xEyOXnGiaAwIoSq1QZ/SS4AhjoAh9DnfY= +github.com/hashicorp/go-retryablehttp v0.6.4/go.mod h1:vAew36LZh98gCBJNLH42IQ1ER/9wtLZZ8meHqQvEYWY= github.com/juju/collections v0.0.0-20180515203731-520e0549d51a h1:PPCCWrZzJMhFu4PxX3vRM65dq7LZMyreWAMPsvttUQk= github.com/juju/collections v0.0.0-20180515203731-520e0549d51a/go.mod h1:Ep+c0vnxsgmmTtsMibPgEEleZyi0b4uVvyzJ+8ka9EI= github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 h1:Sem5Flzxj8ZdAgY2wfHBUlOYyP4wrpIfM8IZgANNGh8= github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18/go.mod h1:W54LbzXuIE0boCoNJfwqpmkKJ1O4TCTZMetAt6jGk7Q= +github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba h1:bnsLBCH2BUzTrl+XckuyTypG0AIWSAALekr1Rvqiy0Q= +github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba/go.mod h1:ppx9XlnQMX/+h/kH+cU9kfDUT6GimqGtNRWdobUZVRE= github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 h1:Y+lzErDTURqeXqlqYi4YBYbDd7ycU74gW1ADt57/bgY= github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U= +github.com/juju/retry v0.0.0-20151029024821-62c620325291 h1:Rp0pLxDOsLDDwh2S73oHLI2KTFFyrF6oM/DgP0FhhBk= github.com/juju/retry v0.0.0-20151029024821-62c620325291/go.mod h1:OohPQGsr4pnxwD5YljhQ+TZnuVRYpa5irjugL1Yuif4= github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d h1:JYANSZLNBXFgnNfGDOUAV+atWFDmOqJ1WPNmyS+YCCw= github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d/go.mod h1:7dL+43wADDfx5rD9ibr5H9Dgr4iOM3uHOa1i4IVLak8= +github.com/juju/testing v0.0.0-20180402130637-44801989f0f7 h1:IOzyKRl+7X8/fDIqNUDQH73yo8bqDrMEh90y9Il158A= github.com/juju/testing v0.0.0-20180402130637-44801989f0f7/go.mod h1:63prj8cnj0tU0S9OHjGJn+b1h0ZghCndfnbQolrYTwA= +github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043 h1:kjdsJcIYzmK2k4X2yVCi5Nip6sGoAuc7CLbp+qQnQUM= github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043/go.mod h1:6/KLg8Wz/y2KVGWEpkK9vMNGkOnu4k/cqs8Z1fKjTOk= github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2 h1:loQDi5MyxxNm7Q42mBGuPD6X+F6zw8j5S9yexLgn/BE= github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2/go.mod h1:kE8gK5X0CImdr7qpSKl3xB2PmpySSmfj7zVbkZFs81U= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4 h1:OfaUle5HH9Y0obNU74mlOZ/Igdtwi3eGOKcljJsTnbw= golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/net v0.0.0-20180406214816-61147c48b25b h1:7rskAFQwNXGW6AD8E/6y0LDHW5mT9rsLD7ViLVFfh5w= golang.org/x/net v0.0.0-20180406214816-61147c48b25b/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2 h1:+j1SppRob9bAgoYmsdW9NNBdKZfgYuWpqnYHv78Qt8w= gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 h1:hILp2hNrRnYjZpmIbx70psAHbBSEcQ1NIzDcUbJ1b6g= gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4/go.mod h1:yeKp02qBN3iKW1OzL3MGk2IdtZzaj7SFntXj72NppTA= +gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 h1:CvAnnm1XvMjfib69SZzDwgWfOk+PxYz0hA0HBupilBA= gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= diff --git a/oauth.go b/oauth.go index 920960d..75a095d 100644 --- a/oauth.go +++ b/oauth.go @@ -28,7 +28,7 @@ func generateTimestamp() string { } type OAuthSigner interface { - OAuthSign(request *http.Request) error + OAuthSign(headers *http.Header) error } type OAuthToken struct { @@ -52,7 +52,7 @@ func NewPlainTestOAuthSigner(token *OAuthToken, realm string) (OAuthSigner, erro // OAuthSignPLAINTEXT signs the provided request using the OAuth PLAINTEXT // method: http://oauth.net/core/1.0/#anchor22. -func (signer plainTextOAuthSigner) OAuthSign(request *http.Request) error { +func (signer plainTextOAuthSigner) OAuthSign(headers *http.Header) error { signature := signer.token.ConsumerSecret + `&` + signer.token.TokenSecret nonce, err := generateNonce() @@ -75,6 +75,6 @@ func (signer plainTextOAuthSigner) OAuthSign(request *http.Request) error { authHeader = append(authHeader, fmt.Sprintf(`%s="%s"`, key, url.QueryEscape(value))) } strHeader := "OAuth " + strings.Join(authHeader, ", ") - request.Header.Add("Authorization", strHeader) + headers.Add("Authorization", strHeader) return nil } From 9c17aad738b44ca0b57289cd1a55ec068e2aa80e Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 10 Mar 2020 09:41:30 -0700 Subject: [PATCH 14/21] Add machine.Owner --- interfaces.go | 1 + machine.go | 9 +++++++++ 2 files changed, 10 insertions(+) diff --git a/interfaces.go b/interfaces.go index b42420b..4b3d0c4 100644 --- a/interfaces.go +++ b/interfaces.go @@ -219,6 +219,7 @@ type Machine interface { SystemID() string Hostname() string FQDN() string + Owner() string Tags() []string OperatingSystem() string diff --git a/machine.go b/machine.go index 11d922a..64be5c9 100644 --- a/machine.go +++ b/machine.go @@ -22,6 +22,7 @@ type machine struct { systemID string hostname string fqdn string + owner string tags []string ownerData map[string]string @@ -53,6 +54,7 @@ func (m *machine) updateFrom(other *machine) { m.systemID = other.systemID m.hostname = other.hostname m.fqdn = other.fqdn + m.owner = other.owner m.operatingSystem = other.operatingSystem m.distroSeries = other.distroSeries m.architecture = other.architecture @@ -83,6 +85,11 @@ func (m *machine) FQDN() string { return m.fqdn } +// Owner implements Machine. +func (m *machine) Owner() string { + return m.owner +} + // Tags implements Machine. func (m *machine) Tags() []string { return m.tags @@ -667,6 +674,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { "system_id": schema.String(), "hostname": schema.String(), "fqdn": schema.String(), + "owner": schema.String(), "tag_names": schema.List(schema.String()), "owner_data": schema.StringMap(schema.String()), @@ -746,6 +754,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { systemID: valid["system_id"].(string), hostname: valid["hostname"].(string), fqdn: valid["fqdn"].(string), + owner: valid["owner"].(string), tags: convertToStringSlice(valid["tag_names"]), ownerData: convertToStringMap(valid["owner_data"]), From 7dc2b2b5172f6d0fa0df09d87424588626aacc70 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 10 Mar 2020 21:14:59 -0700 Subject: [PATCH 15/21] Add storage device creation functions --- blockdevice.go | 75 +++++++++++++++++++++++-- interfaces.go | 19 +++++++ machine.go | 136 +++++++++++++++++++++++++++++++++++++++++++++ partition.go | 32 +++++++++-- volumegroup.go | 146 +++++++++++++++++++++++++++++++++++++++++++++++++ 5 files changed, 397 insertions(+), 11 deletions(-) create mode 100644 volumegroup.go diff --git a/blockdevice.go b/blockdevice.go index b107635..c83c2cf 100644 --- a/blockdevice.go +++ b/blockdevice.go @@ -4,6 +4,8 @@ package gomaasapi import ( + "net/http" + "github.com/juju/errors" "github.com/juju/schema" "github.com/juju/version" @@ -11,6 +13,7 @@ import ( type blockdevice struct { resourceURI string + controller *controller id int uuid string @@ -103,14 +106,44 @@ func (b *blockdevice) Partitions() []Partition { return result } -func readBlockDevices(controllerVersion version.Number, source interface{}) ([]*blockdevice, error) { - checker := schema.List(schema.StringMap(schema.Any())) - coerced, err := checker.Coerce(source, nil) +// CreatePartitionArgs options for creating partitions +type CreatePartitionArgs struct { + Size int // Optional. The size of the partition. If not specified, all available space will be used. + UUID string // Optional. UUID for the partition. Only used if the partition table type for the block device is GPT. + Bootable bool // Optional. If the partition should be marked bootable. +} + +func (a *CreatePartitionArgs) toParams() *URLParams { + params := &URLParams{} + params.MaybeAddInt("size", a.Size) + params.MaybeAdd("uuid", a.UUID) + params.MaybeAddBool("bootable", a.Bootable) + return params +} + +func (b *blockdevice) CreatePartition(args CreatePartitionArgs) (Partition, error) { + params := args.toParams() + source, err := b.controller.post(b.resourceURI+"/partitions/", "", params.Values) if err != nil { - return nil, WrapWithDeserializationError(err, "blockdevice base schema check failed") + 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) } - valid := coerced.([]interface{}) + response, err := readPartition(b.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + return response, nil +} + +func getBlockDeviceDeserializationFunc(controllerVersion version.Number) (blockdeviceDeserializationFunc, error) { var deserialisationVersion version.Number for v := range blockdeviceDeserializationFuncs { if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 { @@ -120,7 +153,37 @@ func readBlockDevices(controllerVersion version.Number, source interface{}) ([]* if deserialisationVersion == version.Zero { return nil, NewUnsupportedVersionError("no blockdevice read func for version %s", controllerVersion) } - readFunc := blockdeviceDeserializationFuncs[deserialisationVersion] + return blockdeviceDeserializationFuncs[deserialisationVersion], nil +} + +func readBlockDevice(controllerVersion version.Number, source interface{}) (*blockdevice, error) { + readFunc, err := getBlockDeviceDeserializationFunc(controllerVersion) + if err != nil { + return nil, err + } + + checker := schema.StringMap(schema.Any()) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, WrapWithDeserializationError(err, "machine base schema check failed") + } + valid := coerced.(map[string]interface{}) + + return readFunc(valid) +} + +func readBlockDevices(controllerVersion version.Number, source interface{}) ([]*blockdevice, error) { + checker := schema.List(schema.StringMap(schema.Any())) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, WrapWithDeserializationError(err, "blockdevice base schema check failed") + } + valid := coerced.([]interface{}) + + readFunc, err := getBlockDeviceDeserializationFunc(controllerVersion) + if err != nil { + return nil, err + } return readBlockDeviceList(valid, readFunc) } diff --git a/interfaces.go b/interfaces.go index 4b3d0c4..ec47086 100644 --- a/interfaces.go +++ b/interfaces.go @@ -270,10 +270,16 @@ type Machine interface { // id specified. If there is no match, nil is returned. BlockDevice(id int) BlockDevice + // CreateBlockDevice will create a new block device + CreateBlockDevice(args CreateBlockDeviceArgs) (BlockDevice, error) + // Partition returns the partition for the machine that matches the // id specified. If there is no match, nil is returned. Partition(id int) Partition + // VolumeGroups returns all volume groups on the machine + VolumeGroups() []VolumeGroup + Zone() Zone Pool() Pool @@ -414,6 +420,8 @@ type StorageDevice interface { // as a filesystem. type Partition interface { StorageDevice + + Format(FormatPartitionArgs) (Partition, error) } // BlockDevice represents an entire block device on the machine. @@ -429,10 +437,21 @@ type BlockDevice interface { Partitions() []Partition + // CreatePartition creates a partition on the provided block device + CreatePartition(CreatePartitionArgs) (Partition, error) + // There are some other attributes for block devices, but we can // expose them on an as needed basis. } +// VolumeGroup represents a collection of logical volumes +type VolumeGroup interface { + Name() string + Size() uint64 + UUID() string + Devices() []BlockDevice +} + // OwnerDataHolder represents any MAAS object that can store key/value // data. type OwnerDataHolder interface { diff --git a/machine.go b/machine.go index 64be5c9..f18563c 100644 --- a/machine.go +++ b/machine.go @@ -47,6 +47,7 @@ type machine struct { // Don't really know the difference between these two lists: physicalBlockDevices []*blockdevice blockDevices []*blockdevice + volumeGroups []*volumegroup } func (m *machine) updateFrom(other *machine) { @@ -244,6 +245,15 @@ func partitionById(id int, blockDevices []BlockDevice) Partition { return nil } +// BlockDevices implements Machine. +func (m *machine) VolumeGroups() []VolumeGroup { + result := make([]VolumeGroup, len(m.volumeGroups)) + for i, v := range m.volumeGroups { + result[i] = v + } + return result +} + // Devices implements Machine. func (m *machine) Devices(args DevicesArgs) ([]Device, error) { // Perhaps in the future, MAAS will give us a way to query just for the @@ -697,6 +707,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { "physicalblockdevice_set": schema.List(schema.StringMap(schema.Any())), "blockdevice_set": schema.List(schema.StringMap(schema.Any())), + "volume_groups": schema.List(schema.StringMap(schema.Any())), } defaults := schema.Defaults{ "architecture": "", @@ -745,6 +756,12 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { if err != nil { return nil, errors.Trace(err) } + + volumeGroups, err := readVolumeGroupList(valid["volume_groups"].([]interface{}), volumegroup_2_0) + if err != nil { + return nil, errors.Trace(err) + } + architecture, _ := valid["architecture"].(string) statusMessage, _ := valid["status_message"].(string) hweKernel, _ := valid["hwe_kernel"].(string) @@ -776,6 +793,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { pool: pool, physicalBlockDevices: physicalBlockDevices, blockDevices: blockDevices, + volumeGroups: volumeGroups, } return result, nil @@ -806,3 +824,121 @@ func convertToStringMap(field interface{}) map[string]string { } return result } + +// CreateBlockDeviceArgs are required parameters +type CreateBlockDeviceArgs struct { + Name string // Required. Name of the block device. + Model string // Optional. Model of the block device. + Serial string // Optional. Serial number of the block device. + IDPath string // Optional. Only used if model and serial cannot be provided. This should be a path that is fixed and doesn't change depending on the boot order or kernel version. + Size int // Required. Size of the block device. + BlockSize int // Required. Block size of the block device. +} + +// ToParams converts arguments to URL parameters +func (a *CreateBlockDeviceArgs) toParams() *URLParams { + params := &URLParams{} + params.MaybeAdd("name", a.Name) + params.MaybeAdd("model", a.Model) + params.MaybeAdd("serial", a.Serial) + params.MaybeAdd("id_path", a.IDPath) + params.MaybeAddInt("size", a.Size) + params.MaybeAddInt("block_size", a.BlockSize) + return params +} + +// Validate checks for invalid configuration +func (a *CreateBlockDeviceArgs) Validate() error { + if a.Name == "" { + return fmt.Errorf("Name must be provided") + } + if a.Size <= 0 { + return fmt.Errorf("Size must be > 0") + } + if a.BlockSize <= 0 { + return fmt.Errorf("Block size must be > 0") + } + return nil +} + +func (m *machine) nodesURI() string { + return strings.Replace(m.resourceURI, "devices", "nodes", 1) +} + +// CreateBlockDevice implementes Machine +func (m *machine) CreateBlockDevice(args CreateBlockDeviceArgs) (BlockDevice, error) { + if err := args.Validate(); err != nil { + return nil, errors.Trace(err) + } + + params := args.toParams() + source, err := m.controller.post(m.nodesURI()+"blockdevices/", "", params.Values) + 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) + } + + response, err := readBlockDevice(m.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + return response, nil +} + +// CreateVolumeGroupArgs control creation of a volume group +type CreateVolumeGroupArgs struct { + Name string // Required. Name of the volume group. + UUID string // Optional. (optional) UUID of the volume group. + BlockDevices []string // Optional. Block devices to add to the volume group. + Partitions []string // Optional. Partitions to add to the volume group. +} + +func (a *CreateVolumeGroupArgs) toParams() *URLParams { + params := &URLParams{} + params.MaybeAdd("name", a.Name) + params.MaybeAdd("uuid", a.UUID) + if a.BlockDevices != nil { + params.MaybeAdd("block_devices", strings.Join(a.BlockDevices, ",")) + } + if a.Partitions != nil { + params.MaybeAdd("partitions", strings.Join(a.Partitions, ",")) + } + return params +} + +// Validate checks for errors +func (a *CreateVolumeGroupArgs) Validate() error { + if a.Name == "" { + return fmt.Errorf("Name required") + } + return nil +} + +func (m *machine) CreateVolumeGroup(args CreateVolumeGroupArgs) (VolumeGroup, error) { + params := args.toParams() + source, err := m.controller.post(m.nodesURI()+"volume-groups", "", params.Values) + 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) + } + + response, err := readVolumeGroup(m.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + return response, nil +} diff --git a/partition.go b/partition.go index 26401e5..da929bd 100644 --- a/partition.go +++ b/partition.go @@ -65,14 +65,22 @@ func (p *partition) Tags() []string { return p.tags } -func readPartitions(controllerVersion version.Number, source interface{}) ([]*partition, error) { - checker := schema.List(schema.StringMap(schema.Any())) +func readPartition(controllerVersion version.Number, source interface{}) (*partition, error) { + readFunc, err := getPartitionDeserializationFunc(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, "partition base schema check failed") + return nil, WrapWithDeserializationError(err, "machine base schema check failed") } - valid := coerced.([]interface{}) + valid := coerced.(map[string]interface{}) + return readFunc(valid) +} +func getPartitionDeserializationFunc(controllerVersion version.Number) (partitionDeserializationFunc, error) { var deserialisationVersion version.Number for v := range partitionDeserializationFuncs { if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 { @@ -82,7 +90,21 @@ func readPartitions(controllerVersion version.Number, source interface{}) ([]*pa if deserialisationVersion == version.Zero { return nil, NewUnsupportedVersionError("no partition read func for version %s", controllerVersion) } - readFunc := partitionDeserializationFuncs[deserialisationVersion] + return partitionDeserializationFuncs[deserialisationVersion], nil +} + +func readPartitions(controllerVersion version.Number, source interface{}) ([]*partition, error) { + checker := schema.List(schema.StringMap(schema.Any())) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, WrapWithDeserializationError(err, "partition base schema check failed") + } + valid := coerced.([]interface{}) + + readFunc, err := getPartitionDeserializationFunc(controllerVersion) + if err != nil { + return nil, err + } return readPartitionList(valid, readFunc) } diff --git a/volumegroup.go b/volumegroup.go new file mode 100644 index 0000000..a85b9b7 --- /dev/null +++ b/volumegroup.go @@ -0,0 +1,146 @@ +package gomaasapi + +import ( + "github.com/juju/errors" + "github.com/juju/schema" + "github.com/juju/version" +) + +type volumegroup struct { + // Add the controller in when we need to do things with the zone. + // controller Controller + + resourceURI string + + id int + name string + description string + uuid string + size uint64 + devices []*blockdevice +} + +func (vg *volumegroup) Name() string { + return vg.name +} + +func (vg *volumegroup) Size() uint64 { + return vg.size +} + +func (vg *volumegroup) UUID() string { + return vg.uuid +} + +func (vg *volumegroup) Devices() []BlockDevice { + result := make([]BlockDevice, len(vg.devices)) + for i, v := range vg.devices { + //v.controller = d + result[i] = v + } + return result +} + +func readVolumeGroup(controllerVersion version.Number, source interface{}) (*volumegroup, error) { + readFunc, err := getVolumeGroupDeserializationFunc(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, "machine base schema check failed") + } + valid := coerced.(map[string]interface{}) + return readFunc(valid) +} + +func getVolumeGroupDeserializationFunc(controllerVersion version.Number) (volumeGroupDeserializationFunc, error) { + var deserialisationVersion version.Number + for v := range volumeGroupDeserializationFuncs { + if v.Compare(deserialisationVersion) > 0 && v.Compare(controllerVersion) <= 0 { + deserialisationVersion = v + } + } + if deserialisationVersion == version.Zero { + return nil, NewUnsupportedVersionError("no volumegroup read func for version %s", controllerVersion) + } + return volumeGroupDeserializationFuncs[deserialisationVersion], nil +} + +func readVolumeGroups(controllerVersion version.Number, source interface{}) ([]*volumegroup, error) { + checker := schema.List(schema.StringMap(schema.Any())) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, WrapWithDeserializationError(err, "volumegroup base schema check failed") + } + valid := coerced.([]interface{}) + + readFunc, err := getVolumeGroupDeserializationFunc(controllerVersion) + if err != nil { + return nil, err + } + return readVolumeGroupList(valid, readFunc) +} + +// readPartitionList expects the values of the sourceList to be string maps. +func readVolumeGroupList(sourceList []interface{}, readFunc volumeGroupDeserializationFunc) ([]*volumegroup, error) { + result := make([]*volumegroup, 0, len(sourceList)) + for i, value := range sourceList { + source, ok := value.(map[string]interface{}) + if !ok { + return nil, NewDeserializationError("unexpected value for volumegroup %d, %T", i, value) + } + volumegroup, err := readFunc(source) + if err != nil { + return nil, errors.Annotatef(err, "volumegroup %d", i) + } + result = append(result, volumegroup) + } + return result, nil +} + +type volumeGroupDeserializationFunc func(map[string]interface{}) (*volumegroup, error) + +var volumeGroupDeserializationFuncs = map[version.Number]volumeGroupDeserializationFunc{ + twoDotOh: volumegroup_2_0, +} + +func volumegroup_2_0(source map[string]interface{}) (*volumegroup, error) { + fields := schema.Fields{ + "resource_uri": schema.String(), + "id": schema.ForceInt(), + "name": schema.String(), + "uuid": schema.OneOf(schema.Nil(""), schema.String()), + "size": schema.ForceUint(), + "devices": schema.List(schema.StringMap(schema.Any())), + } + defaults := schema.Defaults{ + //"tags": []string{}, + } + checker := schema.FieldMap(fields, defaults) + coerced, err := checker.Coerce(source, nil) + if err != nil { + return nil, WrapWithDeserializationError(err, "volumegroup 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. + + devices, err := readBlockDeviceList(valid["devices"].([]interface{}), blockdevice_2_0) + if err != nil { + return nil, errors.Trace(err) + } + + uuid, _ := valid["uuid"].(string) + result := &volumegroup{ + resourceURI: valid["resource_uri"].(string), + id: valid["id"].(int), + name: valid["name"].(string), + uuid: uuid, + size: valid["size"].(uint64), + devices: devices, + } + return result, nil +} From a26dacec170b4084b46174c7dc74a4d264cbb3f3 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 10 Mar 2020 21:37:32 -0700 Subject: [PATCH 16/21] Add partition.Format --- blockdevice.go | 1 + interfaces.go | 2 +- machine.go | 13 +++++++---- partition.go | 63 ++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 74 insertions(+), 5 deletions(-) diff --git a/blockdevice.go b/blockdevice.go index c83c2cf..6d8d29f 100644 --- a/blockdevice.go +++ b/blockdevice.go @@ -101,6 +101,7 @@ func (b *blockdevice) FileSystem() FileSystem { func (b *blockdevice) Partitions() []Partition { result := make([]Partition, len(b.partitions)) for i, v := range b.partitions { + v.controller = b.controller result[i] = v } return result diff --git a/interfaces.go b/interfaces.go index ec47086..0586e77 100644 --- a/interfaces.go +++ b/interfaces.go @@ -421,7 +421,7 @@ type StorageDevice interface { type Partition interface { StorageDevice - Format(FormatPartitionArgs) (Partition, error) + Format(FormatPartitionArgs) error } // BlockDevice represents an entire block device on the machine. diff --git a/machine.go b/machine.go index f18563c..9eb437e 100644 --- a/machine.go +++ b/machine.go @@ -210,6 +210,7 @@ func (m *machine) PhysicalBlockDevice(id int) BlockDevice { func (m *machine) BlockDevices() []BlockDevice { result := make([]BlockDevice, len(m.blockDevices)) for i, v := range m.blockDevices { + v.controller = m.controller result[i] = v } return result @@ -231,13 +232,17 @@ func blockDeviceById(id int, blockDevices []BlockDevice) BlockDevice { // Partition implements Machine. func (m *machine) Partition(id int) Partition { - return partitionById(id, m.BlockDevices()) + p := partitionById(id, m.blockDevices) + if p != nil { + p.controller = m.controller + } + return p } -func partitionById(id int, blockDevices []BlockDevice) Partition { +func partitionById(id int, blockDevices []*blockdevice) *partition { for _, blockDevice := range blockDevices { - for _, partition := range blockDevice.Partitions() { - if partition.ID() == id { + for _, partition := range blockDevice.partitions { + if partition.id == id { return partition } } diff --git a/partition.go b/partition.go index da929bd..0322469 100644 --- a/partition.go +++ b/partition.go @@ -4,12 +4,16 @@ package gomaasapi import ( + "fmt" + "net/http" + "github.com/juju/errors" "github.com/juju/schema" "github.com/juju/version" ) type partition struct { + controller *controller resourceURI string id int @@ -22,6 +26,17 @@ type partition struct { filesystem *filesystem } +func (p *partition) updateFrom(other *partition) { + p.resourceURI = other.resourceURI + p.id = other.id + p.path = other.path + p.uuid = other.uuid + p.usedFor = other.usedFor + p.size = other.size + p.tags = other.tags + p.filesystem = other.filesystem +} + // Type implements Partition. func (p *partition) Type() string { return "partition" @@ -65,6 +80,54 @@ func (p *partition) Tags() []string { return p.tags } +type FormatPartitionArgs struct { + FSType string // Required. Type of filesystem. + UUID string // Optional. The UUID for the filesystem. + Label string // Optional. The label for the filesystem. +} + +// Validate ensures correct args +func (a *FormatPartitionArgs) Validate() error { + if a.FSType == "" { + return fmt.Errorf("A filesystem type must be specified") + } + + return nil +} + +func (p *partition) Format(args FormatPartitionArgs) error { + if err := args.Validate(); err != nil { + return errors.Trace(err) + } + + params := NewURLParams() + params.MaybeAdd("fs_type", args.FSType) + params.MaybeAdd("uuid", args.UUID) + params.MaybeAdd("label", args.Label) + + result, err := p.controller.post(p.resourceURI, "format", params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound: + return errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return NewUnexpectedError(err) + } + + partition, err := readPartition(p.controller.apiVersion, result) + if err != nil { + return errors.Trace(err) + } + p.updateFrom(partition) + return nil +} + func readPartition(controllerVersion version.Number, source interface{}) (*partition, error) { readFunc, err := getPartitionDeserializationFunc(controllerVersion) if err != nil { From 36ad4c4e486bcdf83e8db46c43d46f15a4316a9f Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 10 Mar 2020 21:53:23 -0700 Subject: [PATCH 17/21] Add volumegroup.CreateLogicalVolume --- interfaces.go | 1 + machine.go | 1 + volumegroup.go | 56 +++++++++++++++++++++++++++++++++++++++++++++++++- 3 files changed, 57 insertions(+), 1 deletion(-) diff --git a/interfaces.go b/interfaces.go index 0586e77..c43e420 100644 --- a/interfaces.go +++ b/interfaces.go @@ -450,6 +450,7 @@ type VolumeGroup interface { Size() uint64 UUID() string Devices() []BlockDevice + CreateLogicalVolume(CreateLogicalVolumeArgs) (BlockDevice, error) } // OwnerDataHolder represents any MAAS object that can store key/value diff --git a/machine.go b/machine.go index 9eb437e..f2db24c 100644 --- a/machine.go +++ b/machine.go @@ -254,6 +254,7 @@ func partitionById(id int, blockDevices []*blockdevice) *partition { func (m *machine) VolumeGroups() []VolumeGroup { result := make([]VolumeGroup, len(m.volumeGroups)) for i, v := range m.volumeGroups { + v.controller = m.controller result[i] = v } return result diff --git a/volumegroup.go b/volumegroup.go index a85b9b7..08448d2 100644 --- a/volumegroup.go +++ b/volumegroup.go @@ -1,6 +1,9 @@ package gomaasapi import ( + "fmt" + "net/http" + "github.com/juju/errors" "github.com/juju/schema" "github.com/juju/version" @@ -8,7 +11,7 @@ import ( type volumegroup struct { // Add the controller in when we need to do things with the zone. - // controller Controller + controller *controller resourceURI string @@ -41,6 +44,57 @@ func (vg *volumegroup) Devices() []BlockDevice { return result } +// CreateLogicalVolumeArgs creates a logical volume in a volume group +type CreateLogicalVolumeArgs struct { + Name string // Required. Name of the logical volume. + UUID string // Optional. (optional) UUID of the logical volume. + Size int // Required. Size of the logical volume. Must be larger than or equal to 4,194,304 bytes. E.g. 4194304. +} + +// Validate ensures arguments are valid +func (a *CreateLogicalVolumeArgs) Validate() error { + if a.Name == "" { + return fmt.Errorf("Name must be specified") + } + if a.Size <= 0 { + return fmt.Errorf("A size > 0 must be specified") + } + return nil +} + +// CreateLogicalVolume creates a logical volume in a volume group +func (vg *volumegroup) CreateLogicalVolume(args CreateLogicalVolumeArgs) (BlockDevice, error) { + if err := args.Validate(); err != nil { + return nil, errors.Trace(err) + } + + params := NewURLParams() + params.MaybeAdd("name", args.Name) + params.MaybeAdd("uuid", args.UUID) + params.MaybeAddInt("size", args.Size) + + result, err := vg.controller.post(vg.resourceURI, "create_logical_volume", params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound: + return nil, errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return nil, errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return nil, errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return nil, NewUnexpectedError(err) + } + + device, err := readBlockDevice(vg.controller.apiVersion, result) + if err != nil { + return nil, errors.Trace(err) + } + return device, nil +} + func readVolumeGroup(controllerVersion version.Number, source interface{}) (*volumegroup, error) { readFunc, err := getVolumeGroupDeserializationFunc(controllerVersion) if err != nil { From 485aefaff6a8a282eea5ebe530f3837aa35d0777 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 10 Mar 2020 22:01:19 -0700 Subject: [PATCH 18/21] Add blockdevice.Format --- blockdevice.go | 66 ++++++++++++++++++++++++++++++++++++++++++++++++++ interfaces.go | 3 +++ 2 files changed, 69 insertions(+) diff --git a/blockdevice.go b/blockdevice.go index 6d8d29f..9b52c6e 100644 --- a/blockdevice.go +++ b/blockdevice.go @@ -4,6 +4,7 @@ package gomaasapi import ( + "fmt" "net/http" "github.com/juju/errors" @@ -32,6 +33,24 @@ type blockdevice struct { partitions []*partition } +func (b *blockdevice) updateFrom(other *blockdevice) { + b.resourceURI = other.resourceURI + b.controller = other.controller + b.id = other.id + b.uuid = other.uuid + b.name = other.name + b.model = other.model + b.idPath = other.idPath + b.path = other.path + b.usedFor = other.usedFor + b.tags = other.tags + b.blockSize = other.blockSize + b.usedSize = other.usedSize + b.size = other.size + b.filesystem = other.filesystem + b.partitions = other.partitions +} + // Type implements BlockDevice func (b *blockdevice) Type() string { return "blockdevice" @@ -107,6 +126,53 @@ func (b *blockdevice) Partitions() []Partition { return result } +// FormatBlockDeviceArgs are options for formatting a block device +type FormatBlockDeviceArgs struct { + FSType string // Required. Type of filesystem. + UUID string // Optional. UUID of the filesystem. +} + +// Validate ensures correct args +func (a *FormatBlockDeviceArgs) Validate() error { + if a.FSType == "" { + return fmt.Errorf("A filesystem type must be specified") + } + + return nil +} + +func (b *blockdevice) Format(args FormatBlockDeviceArgs) error { + if err := args.Validate(); err != nil { + return errors.Trace(err) + } + + params := NewURLParams() + params.MaybeAdd("fs_type", args.FSType) + params.MaybeAdd("uuid", args.UUID) + + result, err := b.controller.post(b.resourceURI, "format", params.Values) + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound: + return errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return NewUnexpectedError(err) + } + + blockDevice, err := readBlockDevice(b.controller.apiVersion, result) + if err != nil { + return errors.Trace(err) + } + b.updateFrom(blockDevice) + return nil +} + // CreatePartitionArgs options for creating partitions type CreatePartitionArgs struct { Size int // Optional. The size of the partition. If not specified, all available space will be used. diff --git a/interfaces.go b/interfaces.go index c43e420..b34da23 100644 --- a/interfaces.go +++ b/interfaces.go @@ -440,6 +440,9 @@ type BlockDevice interface { // CreatePartition creates a partition on the provided block device CreatePartition(CreatePartitionArgs) (Partition, error) + // Format places a filesystem on the block device + Format(FormatBlockDeviceArgs) error + // There are some other attributes for block devices, but we can // expose them on an as needed basis. } From 8f19c6ce7e55c342bc8cd2223a33a3ab0ce0fb59 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Wed, 11 Mar 2020 09:28:26 -0700 Subject: [PATCH 19/21] Fix NPE in volume group reads --- blockdevice.go | 30 +++++++++++++++++++----------- interfaces.go | 4 ++-- machine.go | 46 +++++++++++++++++++++++++++++++--------------- volumegroup.go | 7 +++++++ 4 files changed, 59 insertions(+), 28 deletions(-) diff --git a/blockdevice.go b/blockdevice.go index 9b52c6e..66a1137 100644 --- a/blockdevice.go +++ b/blockdevice.go @@ -283,19 +283,19 @@ func blockdevice_2_0(source map[string]interface{}) (*blockdevice, error) { "id": schema.ForceInt(), "uuid": schema.OneOf(schema.Nil(""), schema.String()), - "name": schema.String(), + "name": schema.OneOf(schema.Nil(""), schema.String()), "model": schema.OneOf(schema.Nil(""), schema.String()), "id_path": schema.OneOf(schema.Nil(""), schema.String()), "path": schema.String(), "used_for": schema.String(), - "tags": schema.List(schema.String()), + "tags": schema.OneOf(schema.Nil(""), schema.List(schema.String())), - "block_size": schema.ForceUint(), - "used_size": schema.ForceUint(), + "block_size": schema.OneOf(schema.Nil(""), schema.ForceUint()), + "used_size": schema.OneOf(schema.Nil(""), schema.ForceUint()), "size": schema.ForceUint(), "filesystem": schema.OneOf(schema.Nil(""), schema.StringMap(schema.Any())), - "partitions": schema.List(schema.StringMap(schema.Any())), + "partitions": schema.OneOf(schema.Nil(""), schema.List(schema.StringMap(schema.Any()))), } checker := schema.FieldMap(fields, nil) coerced, err := checker.Coerce(source, nil) @@ -312,28 +312,36 @@ func blockdevice_2_0(source map[string]interface{}) (*blockdevice, error) { return nil, errors.Trace(err) } } - partitions, err := readPartitionList(valid["partitions"].([]interface{}), partition_2_0) - if err != nil { - return nil, errors.Trace(err) + + partitions := []*partition{} + if valid["partitions"] != nil { + var err error + partitions, err = readPartitionList(valid["partitions"].([]interface{}), partition_2_0) + if err != nil { + return nil, errors.Trace(err) + } } uuid, _ := valid["uuid"].(string) model, _ := valid["model"].(string) idPath, _ := valid["id_path"].(string) + name, _ := valid["name"].(string) + blockSize, _ := valid["block_size"].(uint64) + usedSize, _ := valid["used_size"].(uint64) result := &blockdevice{ resourceURI: valid["resource_uri"].(string), id: valid["id"].(int), uuid: uuid, - name: valid["name"].(string), + name: name, model: model, idPath: idPath, path: valid["path"].(string), usedFor: valid["used_for"].(string), tags: convertToStringSlice(valid["tags"]), - blockSize: valid["block_size"].(uint64), - usedSize: valid["used_size"].(uint64), + blockSize: blockSize, + usedSize: usedSize, size: valid["size"].(uint64), filesystem: filesystem, diff --git a/interfaces.go b/interfaces.go index b34da23..7724cf3 100644 --- a/interfaces.go +++ b/interfaces.go @@ -277,8 +277,8 @@ type Machine interface { // id specified. If there is no match, nil is returned. Partition(id int) Partition - // VolumeGroups returns all volume groups on the machine - VolumeGroups() []VolumeGroup + // VolumeGroups returns all volume groups on the machine (dynamically loaded) + VolumeGroups() ([]VolumeGroup, error) Zone() Zone Pool() Pool diff --git a/machine.go b/machine.go index f2db24c..3e2f4f9 100644 --- a/machine.go +++ b/machine.go @@ -47,7 +47,6 @@ type machine struct { // Don't really know the difference between these two lists: physicalBlockDevices []*blockdevice blockDevices []*blockdevice - volumeGroups []*volumegroup } func (m *machine) updateFrom(other *machine) { @@ -69,6 +68,8 @@ func (m *machine) updateFrom(other *machine) { m.pool = other.pool m.tags = other.tags m.ownerData = other.ownerData + m.physicalBlockDevices = other.physicalBlockDevices + m.blockDevices = other.blockDevices } // SystemID implements Machine. @@ -250,14 +251,34 @@ func partitionById(id int, blockDevices []*blockdevice) *partition { return nil } -// BlockDevices implements Machine. -func (m *machine) VolumeGroups() []VolumeGroup { - result := make([]VolumeGroup, len(m.volumeGroups)) - for i, v := range m.volumeGroups { +// BlockDevices implements Machine (loaded dynamically) +func (m *machine) VolumeGroups() ([]VolumeGroup, error) { + source, err := m.controller.get(m.nodesURI() + "volume-groups/") + if err != nil { + if svrErr, ok := errors.Cause(err).(ServerError); ok { + switch svrErr.StatusCode { + case http.StatusNotFound, http.StatusConflict: + return nil, errors.Wrap(err, NewBadRequestError(svrErr.BodyMessage)) + case http.StatusForbidden: + return nil, errors.Wrap(err, NewPermissionError(svrErr.BodyMessage)) + case http.StatusServiceUnavailable: + return nil, errors.Wrap(err, NewCannotCompleteError(svrErr.BodyMessage)) + } + } + return nil, NewUnexpectedError(err) + } + + vgs, err := readVolumeGroups(m.controller.apiVersion, source) + if err != nil { + return nil, errors.Trace(err) + } + + result := make([]VolumeGroup, len(vgs)) + for i, v := range vgs { v.controller = m.controller result[i] = v } - return result + return result, nil } // Devices implements Machine. @@ -690,7 +711,7 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { "system_id": schema.String(), "hostname": schema.String(), "fqdn": schema.String(), - "owner": schema.String(), + "owner": schema.OneOf(schema.Nil(""), schema.String()), "tag_names": schema.List(schema.String()), "owner_data": schema.StringMap(schema.String()), @@ -763,21 +784,17 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { return nil, errors.Trace(err) } - volumeGroups, err := readVolumeGroupList(valid["volume_groups"].([]interface{}), volumegroup_2_0) - if err != nil { - return nil, errors.Trace(err) - } - architecture, _ := valid["architecture"].(string) statusMessage, _ := valid["status_message"].(string) hweKernel, _ := valid["hwe_kernel"].(string) + owner, _ := valid["owner"].(string) result := &machine{ resourceURI: valid["resource_uri"].(string), systemID: valid["system_id"].(string), hostname: valid["hostname"].(string), fqdn: valid["fqdn"].(string), - owner: valid["owner"].(string), + owner: owner, tags: convertToStringSlice(valid["tag_names"]), ownerData: convertToStringMap(valid["owner_data"]), @@ -799,7 +816,6 @@ func machine_2_0(source map[string]interface{}) (*machine, error) { pool: pool, physicalBlockDevices: physicalBlockDevices, blockDevices: blockDevices, - volumeGroups: volumeGroups, } return result, nil @@ -868,7 +884,7 @@ func (a *CreateBlockDeviceArgs) Validate() error { } func (m *machine) nodesURI() string { - return strings.Replace(m.resourceURI, "devices", "nodes", 1) + return strings.Replace(m.resourceURI, "machines", "nodes", 1) } // CreateBlockDevice implementes Machine diff --git a/volumegroup.go b/volumegroup.go index 08448d2..f86bf74 100644 --- a/volumegroup.go +++ b/volumegroup.go @@ -169,6 +169,13 @@ func volumegroup_2_0(source map[string]interface{}) (*volumegroup, error) { "uuid": schema.OneOf(schema.Nil(""), schema.String()), "size": schema.ForceUint(), "devices": schema.List(schema.StringMap(schema.Any())), + //"used_size": schema.String(), + //"human_size": schema.ForceUint(), + //"system_id": schema.String(), + //"available_size": schema.String(), + //"logical_volumes": schema.String(), + //"human_used_size": schema.String(), + //"human_available_size": schema.String(), } defaults := schema.Defaults{ //"tags": []string{}, From fce7f0d2e318a03184f7f018ee3d7ec62fa45d9a Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Thu, 12 Mar 2020 15:30:36 -0700 Subject: [PATCH 20/21] Move Format into StorageDevice interface --- blockdevice.go | 52 +++++++++++++++++++++++++++++++++++++++++++------- interfaces.go | 14 +++++++++----- machine.go | 25 ++++++++++++++++-------- partition.go | 41 ++++++++++++++++++++++----------------- 4 files changed, 95 insertions(+), 37 deletions(-) diff --git a/blockdevice.go b/blockdevice.go index 66a1137..b353979 100644 --- a/blockdevice.go +++ b/blockdevice.go @@ -126,14 +126,15 @@ func (b *blockdevice) Partitions() []Partition { return result } -// FormatBlockDeviceArgs are options for formatting a block device -type FormatBlockDeviceArgs struct { +// FormatStorageDeviceArgs are options for formatting BlockDevices and Partitions +type FormatStorageDeviceArgs struct { FSType string // Required. Type of filesystem. - UUID string // Optional. UUID of the filesystem. + UUID string // Optional. The UUID for the filesystem. + Label string // Optional. The label for the filesystem, only applies to partitions. } // Validate ensures correct args -func (a *FormatBlockDeviceArgs) Validate() error { +func (a *FormatStorageDeviceArgs) Validate() error { if a.FSType == "" { return fmt.Errorf("A filesystem type must be specified") } @@ -141,7 +142,7 @@ func (a *FormatBlockDeviceArgs) Validate() error { return nil } -func (b *blockdevice) Format(args FormatBlockDeviceArgs) error { +func (b *blockdevice) Format(args FormatStorageDeviceArgs) error { if err := args.Validate(); err != nil { return errors.Trace(err) } @@ -181,7 +182,7 @@ type CreatePartitionArgs struct { } func (a *CreatePartitionArgs) toParams() *URLParams { - params := &URLParams{} + params := NewURLParams() params.MaybeAddInt("size", a.Size) params.MaybeAdd("uuid", a.UUID) params.MaybeAddBool("bootable", a.Bootable) @@ -190,7 +191,7 @@ func (a *CreatePartitionArgs) toParams() *URLParams { func (b *blockdevice) CreatePartition(args CreatePartitionArgs) (Partition, error) { params := args.toParams() - source, err := b.controller.post(b.resourceURI+"/partitions/", "", params.Values) + source, err := b.controller.post(b.resourceURI+"partitions/", "", params.Values) if err != nil { if svrErr, ok := errors.Cause(err).(ServerError); ok { switch svrErr.StatusCode { @@ -207,9 +208,46 @@ func (b *blockdevice) CreatePartition(args CreatePartitionArgs) (Partition, erro if err != nil { return nil, errors.Trace(err) } + response.controller = b.controller return response, nil } +// MountStorageDeviceArgs options for creating partitions +type MountStorageDeviceArgs struct { + MountPoint string // Required. Path on the filesystem to mount. + MountOptions string // Optional. Options to pass to mount(8). +} + +func (a *MountStorageDeviceArgs) toParams() *URLParams { + params := NewURLParams() + params.MaybeAdd("mount_point", a.MountPoint) + params.MaybeAdd("mount_options", a.MountOptions) + return params +} + +func (b *blockdevice) Mount(args MountStorageDeviceArgs) error { + params := args.toParams() + source, err := b.controller.post(b.resourceURI, "mount", params.Values) + 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) + } + + response, err := readBlockDevice(b.controller.apiVersion, source) + if err != nil { + return errors.Trace(err) + } + b.updateFrom(response) + return nil +} + func getBlockDeviceDeserializationFunc(controllerVersion version.Number) (blockdeviceDeserializationFunc, error) { var deserialisationVersion version.Number for v := range blockdeviceDeserializationFuncs { diff --git a/interfaces.go b/interfaces.go index 7724cf3..ac225ca 100644 --- a/interfaces.go +++ b/interfaces.go @@ -280,6 +280,9 @@ type Machine interface { // VolumeGroups returns all volume groups on the machine (dynamically loaded) VolumeGroups() ([]VolumeGroup, error) + // CreateVolumeGroup creates a volume group with the provided block devices and partitions + CreateVolumeGroup(args CreateVolumeGroupArgs) (VolumeGroup, error) + Zone() Zone Pool() Pool @@ -414,14 +417,18 @@ type StorageDevice interface { // FileSystem may be nil if not mounted. FileSystem() FileSystem + + // Format places a filesystem on the block device + Format(FormatStorageDeviceArgs) error + + // Mount mounts device at a specific path + Mount(args MountStorageDeviceArgs) error } // Partition represents a partition of a block device. It may be mounted // as a filesystem. type Partition interface { StorageDevice - - Format(FormatPartitionArgs) error } // BlockDevice represents an entire block device on the machine. @@ -440,9 +447,6 @@ type BlockDevice interface { // CreatePartition creates a partition on the provided block device CreatePartition(CreatePartitionArgs) (Partition, error) - // Format places a filesystem on the block device - Format(FormatBlockDeviceArgs) error - // There are some other attributes for block devices, but we can // expose them on an as needed basis. } diff --git a/machine.go b/machine.go index 3e2f4f9..995f43f 100644 --- a/machine.go +++ b/machine.go @@ -197,6 +197,7 @@ func (m *machine) StatusMessage() string { func (m *machine) PhysicalBlockDevices() []BlockDevice { result := make([]BlockDevice, len(m.physicalBlockDevices)) for i, v := range m.physicalBlockDevices { + v.controller = m.controller result[i] = v } return result @@ -859,7 +860,7 @@ type CreateBlockDeviceArgs struct { // ToParams converts arguments to URL parameters func (a *CreateBlockDeviceArgs) toParams() *URLParams { - params := &URLParams{} + params := NewURLParams() params.MaybeAdd("name", a.Name) params.MaybeAdd("model", a.Model) params.MaybeAdd("serial", a.Serial) @@ -916,21 +917,29 @@ func (m *machine) CreateBlockDevice(args CreateBlockDeviceArgs) (BlockDevice, er // CreateVolumeGroupArgs control creation of a volume group type CreateVolumeGroupArgs struct { - Name string // Required. Name of the volume group. - UUID string // Optional. (optional) UUID of the volume group. - BlockDevices []string // Optional. Block devices to add to the volume group. - Partitions []string // Optional. Partitions to add to the volume group. + Name string // Required. Name of the volume group. + UUID string // Optional. (optional) UUID of the volume group. + BlockDevices []BlockDevice // Optional. Block devices to add to the volume group. + Partitions []Partition // Optional. Partitions to add to the volume group. } func (a *CreateVolumeGroupArgs) toParams() *URLParams { - params := &URLParams{} + params := NewURLParams() params.MaybeAdd("name", a.Name) params.MaybeAdd("uuid", a.UUID) if a.BlockDevices != nil { - params.MaybeAdd("block_devices", strings.Join(a.BlockDevices, ",")) + deviceIDs := []string{} + for _, device := range a.BlockDevices { + deviceIDs = append(deviceIDs, fmt.Sprintf("%d", device.ID())) + } + params.MaybeAdd("block_devices", strings.Join(deviceIDs, ",")) } if a.Partitions != nil { - params.MaybeAdd("partitions", strings.Join(a.Partitions, ",")) + partitionIDs := []string{} + for _, partition := range a.Partitions { + partitionIDs = append(partitionIDs, fmt.Sprintf("%d", partition.ID())) + } + params.MaybeAdd("partitions", strings.Join(partitionIDs, ",")) } return params } diff --git a/partition.go b/partition.go index 0322469..e8708d8 100644 --- a/partition.go +++ b/partition.go @@ -4,7 +4,6 @@ package gomaasapi import ( - "fmt" "net/http" "github.com/juju/errors" @@ -80,22 +79,7 @@ func (p *partition) Tags() []string { return p.tags } -type FormatPartitionArgs struct { - FSType string // Required. Type of filesystem. - UUID string // Optional. The UUID for the filesystem. - Label string // Optional. The label for the filesystem. -} - -// Validate ensures correct args -func (a *FormatPartitionArgs) Validate() error { - if a.FSType == "" { - return fmt.Errorf("A filesystem type must be specified") - } - - return nil -} - -func (p *partition) Format(args FormatPartitionArgs) error { +func (p *partition) Format(args FormatStorageDeviceArgs) error { if err := args.Validate(); err != nil { return errors.Trace(err) } @@ -128,6 +112,29 @@ func (p *partition) Format(args FormatPartitionArgs) error { return nil } +func (p *partition) Mount(args MountStorageDeviceArgs) error { + params := args.toParams() + source, err := p.controller.post(p.resourceURI, "mount", params.Values) + 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) + } + + response, err := readPartition(p.controller.apiVersion, source) + if err != nil { + return errors.Trace(err) + } + p.updateFrom(response) + return nil +} + func readPartition(controllerVersion version.Number, source interface{}) (*partition, error) { readFunc, err := getPartitionDeserializationFunc(controllerVersion) if err != nil { From 565e64bc346c12b5573b524c2d487db1f36526d2 Mon Sep 17 00:00:00 2001 From: Sean Houghton Date: Tue, 31 Mar 2020 10:59:44 -0700 Subject: [PATCH 21/21] Update module --- go.mod | 4 ++-- go.sum | 7 +++++-- 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/go.mod b/go.mod index eee3595..62051a7 100644 --- a/go.mod +++ b/go.mod @@ -1,4 +1,4 @@ -module github.com/seanhoughton/gomaasapi +module github.com/juju/gomaasapi go 1.13 @@ -6,7 +6,6 @@ require ( github.com/hashicorp/go-retryablehttp v0.6.4 github.com/juju/collections v0.0.0-20180515203731-520e0549d51a github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 - github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 github.com/juju/retry v0.0.0-20151029024821-62c620325291 // indirect github.com/juju/schema v0.0.0-20160420044203-075de04f9b7d @@ -18,4 +17,5 @@ require ( gopkg.in/check.v1 v1.0.0-20160105164936-4f90aeace3a2 gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 // indirect + launchpad.net/gocheck v0.0.0-20140225173054-000000000087 // indirect ) diff --git a/go.sum b/go.sum index 1985ad0..8032d1d 100644 --- a/go.sum +++ b/go.sum @@ -1,3 +1,4 @@ +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/hashicorp/go-cleanhttp v0.5.1 h1:dH3aiDG9Jvb5r5+bYHsikaOUIpcM0xvgMXVoDkXMzJM= github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= @@ -9,8 +10,6 @@ github.com/juju/collections v0.0.0-20180515203731-520e0549d51a h1:PPCCWrZzJMhFu4 github.com/juju/collections v0.0.0-20180515203731-520e0549d51a/go.mod h1:Ep+c0vnxsgmmTtsMibPgEEleZyi0b4uVvyzJ+8ka9EI= github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18 h1:Sem5Flzxj8ZdAgY2wfHBUlOYyP4wrpIfM8IZgANNGh8= github.com/juju/errors v0.0.0-20150916125642-1b5e39b83d18/go.mod h1:W54LbzXuIE0boCoNJfwqpmkKJ1O4TCTZMetAt6jGk7Q= -github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba h1:bnsLBCH2BUzTrl+XckuyTypG0AIWSAALekr1Rvqiy0Q= -github.com/juju/gomaasapi v0.0.0-20190826212825-0ab1eb636aba/go.mod h1:ppx9XlnQMX/+h/kH+cU9kfDUT6GimqGtNRWdobUZVRE= github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9 h1:Y+lzErDTURqeXqlqYi4YBYbDd7ycU74gW1ADt57/bgY= github.com/juju/loggo v0.0.0-20170605014607-8232ab8918d9/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U= github.com/juju/retry v0.0.0-20151029024821-62c620325291 h1:Rp0pLxDOsLDDwh2S73oHLI2KTFFyrF6oM/DgP0FhhBk= @@ -23,7 +22,9 @@ github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043 h1:kjdsJcIYzmK2k4X2yVCi github.com/juju/utils v0.0.0-20180424094159-2000ea4ff043/go.mod h1:6/KLg8Wz/y2KVGWEpkK9vMNGkOnu4k/cqs8Z1fKjTOk= github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2 h1:loQDi5MyxxNm7Q42mBGuPD6X+F6zw8j5S9yexLgn/BE= github.com/juju/version v0.0.0-20161031051906-1f41e27e54f2/go.mod h1:kE8gK5X0CImdr7qpSKl3xB2PmpySSmfj7zVbkZFs81U= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4 h1:OfaUle5HH9Y0obNU74mlOZ/Igdtwi3eGOKcljJsTnbw= golang.org/x/crypto v0.0.0-20180214000028-650f4a345ab4/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= @@ -35,3 +36,5 @@ gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4 h1:hILp2hNrRnYjZpmIbx70psAHbB gopkg.in/mgo.v2 v2.0.0-20160818015218-f2b6f6c918c4/go.mod h1:yeKp02qBN3iKW1OzL3MGk2IdtZzaj7SFntXj72NppTA= gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6 h1:CvAnnm1XvMjfib69SZzDwgWfOk+PxYz0hA0HBupilBA= gopkg.in/yaml.v2 v2.0.0-20170712054546-1be3d31502d6/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= +launchpad.net/gocheck v0.0.0-20140225173054-000000000087 h1:Izowp2XBH6Ya6rv+hqbceQyw/gSGoXfH/UPoTGduL54= +launchpad.net/gocheck v0.0.0-20140225173054-000000000087/go.mod h1:hj7XX3B/0A+80Vse0e+BUHsHMTEhd0O4cpUHr/e/BUM=