// constructImageConstraint returns model-specific criteria used to look for image metadata. func (p *ProvisionerAPI) constructImageConstraint(m *state.Machine) (*imagemetadata.ImageConstraint, environs.Environ, error) { // If we can determine current region, // we want only metadata specific to this region. cloud, env, err := p.obtainEnvCloudConfig() if err != nil { return nil, nil, errors.Trace(err) } lookup := simplestreams.LookupParams{ Series: []string{m.Series()}, Stream: env.Config().ImageStream(), } mcons, err := m.Constraints() if err != nil { return nil, nil, errors.Annotatef(err, "cannot get machine constraints for machine %v", m.MachineTag().Id()) } if mcons.Arch != nil { lookup.Arches = []string{*mcons.Arch} } if cloud != nil { lookup.CloudSpec = *cloud } return imagemetadata.NewImageConstraint(lookup), env, nil }
func getProvisioningInfo(m *state.Machine) (*params.ProvisioningInfo, error) { cons, err := m.Constraints() if err != nil { return nil, err } // TODO(dimitern) For now, since network names and // provider ids are the same, we return what we got // from state. In the future, when networks can be // added before provisioning, we should convert both // slices from juju network names to provider-specific // ids before returning them. networks, err := m.RequestedNetworks() if err != nil { return nil, err } var jobs []params.MachineJob for _, job := range m.Jobs() { jobs = append(jobs, job.ToParams()) } return ¶ms.ProvisioningInfo{ Constraints: cons, Series: m.Series(), Placement: m.Placement(), Networks: networks, Jobs: jobs, }, nil }
// machineSubnetsAndZones returns a map of subnet provider-specific id // to list of availability zone names for that subnet. The result can // be empty if there are no spaces constraints specified for the // machine, or there's an error fetching them. func (p *ProvisionerAPI) machineSubnetsAndZones(m *state.Machine) (map[string][]string, error) { mcons, err := m.Constraints() if err != nil { return nil, errors.Annotate(err, "cannot get machine constraints") } includeSpaces := mcons.IncludeSpaces() if len(includeSpaces) < 1 { // Nothing to do. return nil, nil } // TODO(dimitern): For the network model MVP we only use the first // included space and ignore the rest. // // LKK Card: https://canonical.leankit.com/Boards/View/101652562/117352306 // LP Bug: http://pad.lv/1498232 spaceName := includeSpaces[0] if len(includeSpaces) > 1 { logger.Debugf( "using space %q from constraints for machine %q (ignoring remaining: %v)", spaceName, m.Id(), includeSpaces[1:], ) } space, err := p.st.Space(spaceName) if err != nil { return nil, errors.Trace(err) } subnets, err := space.Subnets() if err != nil { return nil, errors.Trace(err) } if len(subnets) == 0 { return nil, errors.Errorf("cannot use space %q as deployment target: no subnets", spaceName) } subnetsToZones := make(map[string][]string, len(subnets)) for _, subnet := range subnets { warningPrefix := fmt.Sprintf( "not using subnet %q in space %q for machine %q provisioning: ", subnet.CIDR(), spaceName, m.Id(), ) providerId := subnet.ProviderId() if providerId == "" { logger.Warningf(warningPrefix + "no ProviderId set") continue } // TODO(dimitern): Once state.Subnet supports multiple zones, // use all of them below. // // LKK Card: https://canonical.leankit.com/Boards/View/101652562/119979611 zone := subnet.AvailabilityZone() if zone == "" { logger.Warningf(warningPrefix + "no availability zone(s) set") continue } subnetsToZones[string(providerId)] = []string{zone} } return subnetsToZones, nil }
func (p *ProvisionerAPI) getProvisioningInfo(m *state.Machine) (*params.ProvisioningInfo, error) { cons, err := m.Constraints() if err != nil { return nil, err } volumes, err := p.machineVolumeParams(m) if err != nil { return nil, errors.Trace(err) } // TODO(dimitern) Drop this once we only use spaces for // deployments. networks, err := m.RequestedNetworks() if err != nil { return nil, err } var jobs []multiwatcher.MachineJob for _, job := range m.Jobs() { jobs = append(jobs, job.ToParams()) } tags, err := p.machineTags(m, jobs) if err != nil { return nil, errors.Trace(err) } subnetsToZones, err := p.machineSubnetsAndZones(m) if err != nil { return nil, errors.Annotate(err, "cannot match subnets to zones") } endpointBindings, err := p.machineEndpointBindings(m) if err != nil { return nil, errors.Annotate(err, "cannot determine machine endpoint bindings") } imageMetadata, err := p.availableImageMetadata(m) if err != nil { return nil, errors.Annotate(err, "cannot get available image metadata") } return ¶ms.ProvisioningInfo{ Constraints: cons, Series: m.Series(), Placement: m.Placement(), Networks: networks, Jobs: jobs, Volumes: volumes, Tags: tags, SubnetsToZones: subnetsToZones, EndpointBindings: endpointBindings, ImageMetadata: imageMetadata, }, nil }
func (p *ProvisionerAPI) getProvisioningInfo(m *state.Machine) (*params.ProvisioningInfo, error) { cons, err := m.Constraints() if err != nil { return nil, errors.Trace(err) } volumes, err := p.machineVolumeParams(m) if err != nil { return nil, errors.Trace(err) } var jobs []multiwatcher.MachineJob for _, job := range m.Jobs() { jobs = append(jobs, job.ToParams()) } tags, err := p.machineTags(m, jobs) if err != nil { return nil, errors.Trace(err) } subnetsToZones, err := p.machineSubnetsAndZones(m) if err != nil { return nil, errors.Annotate(err, "cannot match subnets to zones") } endpointBindings, err := p.machineEndpointBindings(m) if err != nil { return nil, errors.Annotate(err, "cannot determine machine endpoint bindings") } imageMetadata, err := p.availableImageMetadata(m) if err != nil { return nil, errors.Annotate(err, "cannot get available image metadata") } controllerCfg, err := p.st.ControllerConfig() if err != nil { return nil, errors.Annotate(err, "cannot get controller configuration") } return ¶ms.ProvisioningInfo{ Constraints: cons, Series: m.Series(), Placement: m.Placement(), Jobs: jobs, Volumes: volumes, Tags: tags, SubnetsToZones: subnetsToZones, EndpointBindings: endpointBindings, ImageMetadata: imageMetadata, ControllerConfig: controllerCfg, }, nil }
// machineSubnetsAndZones returns a map of subnet provider-specific id // to list of availability zone names for that subnet. The result can // be empty if there are no spaces constraints specified for the // machine, or there's an error fetching them. func (p *ProvisionerAPI) machineSubnetsAndZones(m *state.Machine) (map[string][]string, error) { mcons, err := m.Constraints() if err != nil { return nil, errors.Annotate(err, "cannot get machine constraints") } includeSpaces := mcons.IncludeSpaces() if len(includeSpaces) < 1 { // Nothing to do. return nil, nil } // TODO(dimitern): For the network model MVP we only use the first // included space and ignore the rest. spaceName := includeSpaces[0] if len(includeSpaces) > 1 { logger.Debugf( "using space %q from constraints for machine %q (ignoring remaining: %v)", spaceName, m.Id(), includeSpaces[1:], ) } space, err := p.st.Space(spaceName) if err != nil { return nil, errors.Trace(err) } subnets, err := space.Subnets() if err != nil { return nil, errors.Trace(err) } subnetsToZones := make(map[string][]string, len(subnets)) for _, subnet := range subnets { warningPrefix := fmt.Sprintf( "not using subnet %q in space %q for machine %q provisioning: ", subnet.CIDR(), spaceName, m.Id(), ) // TODO(dimitern): state.Subnet.ProviderId needs to be of type // network.Id. providerId := subnet.ProviderId() if providerId == "" { logger.Warningf(warningPrefix + "no ProviderId set") continue } // TODO(dimitern): Once state.Subnet supports multiple zones, // use all of them below. zone := subnet.AvailabilityZone() if zone == "" { logger.Warningf(warningPrefix + "no availability zone(s) set") continue } subnetsToZones[providerId] = []string{zone} } return subnetsToZones, nil }