mirror of https://github.com/pulumi/pulumi.git
116 lines
4.1 KiB
Go
116 lines
4.1 KiB
Go
// Copyright 2016-2018, Pulumi Corporation.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// Package stack contains the serialized and configurable state associated with an stack; or, in other
|
|
// words, a deployment target. It pertains to resources and deployment plans, but is a package unto itself.
|
|
package stack
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/pulumi/pulumi/pkg/apitype"
|
|
"github.com/pulumi/pulumi/pkg/apitype/migrate"
|
|
"github.com/pulumi/pulumi/pkg/resource"
|
|
"github.com/pulumi/pulumi/pkg/resource/config"
|
|
"github.com/pulumi/pulumi/pkg/resource/deploy"
|
|
"github.com/pulumi/pulumi/pkg/tokens"
|
|
"github.com/pulumi/pulumi/pkg/util/contract"
|
|
)
|
|
|
|
func UnmarshalVersionedCheckpointToLatestCheckpoint(bytes []byte) (*apitype.CheckpointV2, error) {
|
|
var versionedCheckpoint apitype.VersionedCheckpoint
|
|
if err := json.Unmarshal(bytes, &versionedCheckpoint); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
switch versionedCheckpoint.Version {
|
|
case 0:
|
|
// The happens when we are loading a checkpoint file from before we started to version things. Go's
|
|
// json package did not support strict marshalling before 1.10, and we use 1.9 in our toolchain today.
|
|
// After we upgrade, we could consider rewriting this code to use DisallowUnknownFields() on the decoder
|
|
// to have the old checkpoint not even deserialize as an apitype.VersionedCheckpoint.
|
|
var v1checkpoint apitype.CheckpointV1
|
|
if err := json.Unmarshal(bytes, &v1checkpoint); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
checkpoint := migrate.UpToCheckpointV2(v1checkpoint)
|
|
return &checkpoint, nil
|
|
case 1:
|
|
var v1checkpoint apitype.CheckpointV1
|
|
if err := json.Unmarshal(versionedCheckpoint.Checkpoint, &v1checkpoint); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
checkpoint := migrate.UpToCheckpointV2(v1checkpoint)
|
|
return &checkpoint, nil
|
|
case 2:
|
|
var v2checkpoint apitype.CheckpointV2
|
|
if err := json.Unmarshal(versionedCheckpoint.Checkpoint, &v2checkpoint); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &v2checkpoint, nil
|
|
default:
|
|
return nil, errors.Errorf("unsupported checkpoint version %d", versionedCheckpoint.Version)
|
|
}
|
|
}
|
|
|
|
// SerializeCheckpoint turns a snapshot into a data structure suitable for serialization.
|
|
func SerializeCheckpoint(stack tokens.QName, config config.Map, snap *deploy.Snapshot) *apitype.VersionedCheckpoint {
|
|
// If snap is nil, that's okay, we will just create an empty deployment; otherwise, serialize the whole snapshot.
|
|
var latest *apitype.DeploymentV2
|
|
if snap != nil {
|
|
latest = SerializeDeployment(snap)
|
|
}
|
|
|
|
b, err := json.Marshal(apitype.CheckpointV2{
|
|
Stack: stack,
|
|
Config: config,
|
|
Latest: latest,
|
|
})
|
|
contract.AssertNoError(err)
|
|
|
|
return &apitype.VersionedCheckpoint{
|
|
Version: apitype.DeploymentSchemaVersionCurrent,
|
|
Checkpoint: json.RawMessage(b),
|
|
}
|
|
}
|
|
|
|
// DeserializeCheckpoint takes a serialized deployment record and returns its associated snapshot. Returns nil
|
|
// if there have been no deployments performed on this checkpoint.
|
|
func DeserializeCheckpoint(chkpoint *apitype.CheckpointV2) (*deploy.Snapshot, error) {
|
|
contract.Require(chkpoint != nil, "chkpoint")
|
|
if chkpoint.Latest != nil {
|
|
return DeserializeDeploymentV2(*chkpoint.Latest)
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
// GetRootStackResource returns the root stack resource from a given snapshot, or nil if not found. If the stack
|
|
// exists, its output properties, if any, are also returned in the resulting map.
|
|
func GetRootStackResource(snap *deploy.Snapshot) (*resource.State, map[string]interface{}) {
|
|
if snap != nil {
|
|
for _, res := range snap.Resources {
|
|
if res.Type == resource.RootStackType {
|
|
return res, SerializeResource(res).Outputs
|
|
}
|
|
}
|
|
}
|
|
return nil, nil
|
|
}
|