2019-03-26 08:29:15 -04:00
|
|
|
package yandex
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-09-10 10:52:42 -04:00
|
|
|
"fmt"
|
2019-03-26 08:29:15 -04:00
|
|
|
"log"
|
2019-09-23 14:03:17 -04:00
|
|
|
"time"
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-09-23 14:03:17 -04:00
|
|
|
grpc_middleware "github.com/grpc-ecosystem/go-grpc-middleware"
|
2019-03-26 08:29:15 -04:00
|
|
|
"github.com/hashicorp/packer/helper/useragent"
|
|
|
|
"github.com/hashicorp/packer/packer"
|
|
|
|
"google.golang.org/grpc"
|
2019-09-23 14:03:17 -04:00
|
|
|
"google.golang.org/grpc/codes"
|
2019-03-26 08:29:15 -04:00
|
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
|
|
|
|
"github.com/yandex-cloud/go-genproto/yandex/cloud/compute/v1"
|
|
|
|
"github.com/yandex-cloud/go-genproto/yandex/cloud/endpoint"
|
2019-04-09 10:46:41 -04:00
|
|
|
"github.com/yandex-cloud/go-genproto/yandex/cloud/vpc/v1"
|
2019-03-26 08:29:15 -04:00
|
|
|
ycsdk "github.com/yandex-cloud/go-sdk"
|
|
|
|
"github.com/yandex-cloud/go-sdk/iamkey"
|
|
|
|
"github.com/yandex-cloud/go-sdk/pkg/requestid"
|
2019-09-23 14:03:17 -04:00
|
|
|
"github.com/yandex-cloud/go-sdk/pkg/retry"
|
2019-09-10 10:52:42 -04:00
|
|
|
"github.com/yandex-cloud/go-sdk/sdkresolvers"
|
2019-03-26 08:29:15 -04:00
|
|
|
)
|
|
|
|
|
2019-09-23 14:03:17 -04:00
|
|
|
const (
|
|
|
|
defaultExponentialBackoffBase = 50 * time.Millisecond
|
|
|
|
defaultExponentialBackoffCap = 1 * time.Minute
|
|
|
|
)
|
|
|
|
|
2019-03-26 08:29:15 -04:00
|
|
|
type driverYC struct {
|
|
|
|
sdk *ycsdk.SDK
|
|
|
|
ui packer.Ui
|
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func NewDriverYC(ui packer.Ui, config *Config) (Driver, error) {
|
|
|
|
log.Printf("[INFO] Initialize Yandex.Cloud client...")
|
|
|
|
|
|
|
|
sdkConfig := ycsdk.Config{}
|
|
|
|
|
|
|
|
if config.Endpoint != "" {
|
|
|
|
sdkConfig.Endpoint = config.Endpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
2020-06-17 05:41:11 -04:00
|
|
|
case config.Token == "" && config.ServiceAccountKeyFile == "":
|
|
|
|
log.Printf("[INFO] Use Instance Service Account for authentication")
|
|
|
|
sdkConfig.Credentials = ycsdk.InstanceServiceAccount()
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
case config.Token != "":
|
2020-06-17 05:41:11 -04:00
|
|
|
log.Printf("[INFO] Use OAuth token for authentication")
|
2019-04-09 10:46:41 -04:00
|
|
|
sdkConfig.Credentials = ycsdk.OAuthToken(config.Token)
|
|
|
|
|
|
|
|
case config.ServiceAccountKeyFile != "":
|
2020-06-17 05:41:11 -04:00
|
|
|
log.Printf("[INFO] Use Service Account key file %q for authentication", config.ServiceAccountKeyFile)
|
2019-04-09 10:46:41 -04:00
|
|
|
key, err := iamkey.ReadFromJSONFile(config.ServiceAccountKeyFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
credentials, err := ycsdk.ServiceAccountKey(key)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
sdkConfig.Credentials = credentials
|
|
|
|
}
|
|
|
|
|
2019-09-23 14:03:17 -04:00
|
|
|
requestIDInterceptor := requestid.Interceptor()
|
|
|
|
|
|
|
|
retryInterceptor := retry.Interceptor(
|
|
|
|
retry.WithMax(config.MaxRetries),
|
|
|
|
retry.WithCodes(codes.Unavailable),
|
|
|
|
retry.WithAttemptHeader(true),
|
|
|
|
retry.WithBackoff(retry.BackoffExponentialWithJitter(defaultExponentialBackoffBase, defaultExponentialBackoffCap)))
|
|
|
|
|
|
|
|
// Make sure retry interceptor is above id interceptor.
|
|
|
|
// Now we will have new request id for every retry attempt.
|
|
|
|
interceptorChain := grpc_middleware.ChainUnaryClient(retryInterceptor, requestIDInterceptor)
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
userAgentMD := metadata.Pairs("user-agent", useragent.String())
|
|
|
|
|
|
|
|
sdk, err := ycsdk.Build(context.Background(), sdkConfig,
|
|
|
|
grpc.WithDefaultCallOptions(grpc.Header(&userAgentMD)),
|
2019-09-23 14:03:17 -04:00
|
|
|
grpc.WithUnaryInterceptor(interceptorChain))
|
2019-04-09 10:46:41 -04:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err = sdk.ApiEndpoint().ApiEndpoint().List(context.Background(), &endpoint.ListApiEndpointsRequest{}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &driverYC{
|
|
|
|
sdk: sdk,
|
|
|
|
ui: ui,
|
|
|
|
}, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-03-26 08:29:15 -04:00
|
|
|
func (d *driverYC) GetImage(imageID string) (*Image, error) {
|
|
|
|
image, err := d.sdk.Compute().Image().Get(context.Background(), &compute.GetImageRequest{
|
|
|
|
ImageId: imageID,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Image{
|
|
|
|
ID: image.Id,
|
|
|
|
Labels: image.Labels,
|
|
|
|
Licenses: image.ProductIds,
|
|
|
|
Name: image.Name,
|
2020-07-08 18:46:05 -04:00
|
|
|
Description: image.Description,
|
2019-03-26 08:29:15 -04:00
|
|
|
FolderID: image.FolderId,
|
|
|
|
MinDiskSizeGb: toGigabytes(image.MinDiskSize),
|
|
|
|
SizeGb: toGigabytes(image.StorageSize),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2019-04-04 09:17:51 -04:00
|
|
|
func (d *driverYC) GetImageFromFolder(ctx context.Context, folderID string, family string) (*Image, error) {
|
|
|
|
image, err := d.sdk.Compute().Image().GetLatestByFamily(ctx, &compute.GetImageLatestByFamilyRequest{
|
2019-03-26 08:29:15 -04:00
|
|
|
FolderId: folderID,
|
|
|
|
Family: family,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Image{
|
|
|
|
ID: image.Id,
|
|
|
|
Labels: image.Labels,
|
|
|
|
Licenses: image.ProductIds,
|
|
|
|
Name: image.Name,
|
2020-07-08 18:46:05 -04:00
|
|
|
Description: image.Description,
|
2019-03-26 08:29:15 -04:00
|
|
|
FolderID: image.FolderId,
|
2019-04-09 10:46:41 -04:00
|
|
|
Family: image.Family,
|
2019-03-26 08:29:15 -04:00
|
|
|
MinDiskSizeGb: toGigabytes(image.MinDiskSize),
|
|
|
|
SizeGb: toGigabytes(image.StorageSize),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2019-09-10 10:52:42 -04:00
|
|
|
func (d *driverYC) GetImageFromFolderByName(ctx context.Context, folderID string, imageName string) (*Image, error) {
|
|
|
|
imageResolver := sdkresolvers.ImageResolver(imageName, sdkresolvers.FolderID(folderID))
|
|
|
|
|
|
|
|
if err := d.sdk.Resolve(ctx, imageResolver); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to resolve image name: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return d.GetImage(imageResolver.ID())
|
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func (d *driverYC) DeleteImage(ID string) error {
|
|
|
|
ctx := context.TODO()
|
|
|
|
op, err := d.sdk.WrapOperation(d.sdk.Compute().Image().Delete(ctx, &compute.DeleteImageRequest{
|
|
|
|
ImageId: ID,
|
|
|
|
}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
err = op.Wait(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-03-26 08:29:15 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
_, err = op.Response()
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func (d *driverYC) SDK() *ycsdk.SDK {
|
|
|
|
return d.sdk
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func (d *driverYC) DeleteInstance(ctx context.Context, instanceID string) error {
|
|
|
|
op, err := d.sdk.WrapOperation(d.sdk.Compute().Instance().Delete(ctx, &compute.DeleteInstanceRequest{
|
|
|
|
InstanceId: instanceID,
|
|
|
|
}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
err = op.Wait(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-03-26 08:29:15 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
_, err = op.Response()
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func (d *driverYC) DeleteSubnet(ctx context.Context, subnetID string) error {
|
|
|
|
op, err := d.sdk.WrapOperation(d.sdk.VPC().Subnet().Delete(ctx, &vpc.DeleteSubnetRequest{
|
|
|
|
SubnetId: subnetID,
|
|
|
|
}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
err = op.Wait(ctx)
|
2019-03-26 08:29:15 -04:00
|
|
|
if err != nil {
|
2019-04-09 10:46:41 -04:00
|
|
|
return err
|
2019-03-26 08:29:15 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
_, err = op.Response()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *driverYC) DeleteNetwork(ctx context.Context, networkID string) error {
|
|
|
|
op, err := d.sdk.WrapOperation(d.sdk.VPC().Network().Delete(ctx, &vpc.DeleteNetworkRequest{
|
|
|
|
NetworkId: networkID,
|
|
|
|
}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-03-26 08:29:15 -04:00
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
err = op.Wait(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-03-26 08:29:15 -04:00
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
_, err = op.Response()
|
|
|
|
return err
|
2019-03-26 08:29:15 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-04-09 10:46:41 -04:00
|
|
|
func (d *driverYC) DeleteDisk(ctx context.Context, diskID string) error {
|
|
|
|
op, err := d.sdk.WrapOperation(d.sdk.Compute().Disk().Delete(ctx, &compute.DeleteDiskRequest{
|
|
|
|
DiskId: diskID,
|
|
|
|
}))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = op.Wait(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = op.Response()
|
|
|
|
return err
|
2020-04-26 19:19:08 -04:00
|
|
|
}
|
|
|
|
|
2020-04-26 19:37:10 -04:00
|
|
|
func (d *driverYC) GetInstanceMetadata(ctx context.Context, instanceID string, key string) (string, error) {
|
2020-04-26 19:19:08 -04:00
|
|
|
instance, err := d.sdk.Compute().Instance().Get(ctx, &compute.GetInstanceRequest{
|
2020-04-26 19:37:10 -04:00
|
|
|
InstanceId: instanceID,
|
2020-04-26 19:19:08 -04:00
|
|
|
View: compute.InstanceView_FULL,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range instance.GetMetadata() {
|
|
|
|
if k == key {
|
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
}
|
2019-04-09 10:46:41 -04:00
|
|
|
|
2020-04-26 19:19:08 -04:00
|
|
|
return "", fmt.Errorf("Instance metadata key, %s, not found.", key)
|
2019-03-26 08:29:15 -04:00
|
|
|
}
|