This commit will implement the core business logic of `pulumi policy publish` -- code to boot an analyzer, ask it for metadata about the policies it contains, pack the code, and transmit all of this to the Pulumi service.
113 lines
4.6 KiB
Go
113 lines
4.6 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 client
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"net/url"
|
|
"path"
|
|
|
|
"github.com/gorilla/mux"
|
|
)
|
|
|
|
// cleanPath returns the canonical path for p, eliminating . and .. elements.
|
|
// Borrowed from gorilla/mux.
|
|
func cleanPath(p string) string {
|
|
if p == "" {
|
|
return "/"
|
|
}
|
|
|
|
if p[0] != '/' {
|
|
p = "/" + p
|
|
}
|
|
np := path.Clean(p)
|
|
|
|
// path.Clean removes trailing slash except for root;
|
|
// put the trailing slash back if necessary.
|
|
if p[len(p)-1] == '/' && np != "/" {
|
|
np += "/"
|
|
}
|
|
|
|
return np
|
|
}
|
|
|
|
// getEndpoint gets the friendly name of the endpoint with the given method and path.
|
|
func getEndpointName(method, path string) string {
|
|
path = cleanPath(path)
|
|
|
|
u, err := url.Parse("http://localhost" + path)
|
|
if err != nil {
|
|
return "unknown"
|
|
}
|
|
|
|
req := http.Request{
|
|
Method: method,
|
|
URL: u,
|
|
}
|
|
var match mux.RouteMatch
|
|
if !routes.Match(&req, &match) {
|
|
return "unknown"
|
|
}
|
|
|
|
return fmt.Sprintf("api/%s", match.Route.GetName())
|
|
}
|
|
|
|
// routes is the canonical muxer we use to determine friendly names for Pulumi APIs.
|
|
var routes *mux.Router
|
|
|
|
// nolint: lll
|
|
func init() {
|
|
routes = mux.NewRouter()
|
|
|
|
// addEndpoint registers the endpoint with the indicated method, path, and friendly name with the route table.
|
|
// We use this to provide more user-friendly names for the endpoints for annotating trace logs.
|
|
addEndpoint := func(method, path, name string) {
|
|
routes.Path(path).Methods(method).Name(name)
|
|
}
|
|
|
|
addEndpoint("GET", "/api/user", "getCurrentUser")
|
|
addEndpoint("GET", "/api/user/stacks", "listUserStacks")
|
|
addEndpoint("GET", "/api/stacks/{orgName}", "listOrganizationStacks")
|
|
addEndpoint("POST", "/api/stacks/{orgName}", "createStack")
|
|
addEndpoint("DELETE", "/api/stacks/{orgName}/{projectName}/{stackName}", "deleteStack")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}", "getStack")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/export", "exportStack")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/import", "importStack")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/encrypt", "encryptValue")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/decrypt", "decryptValue")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/logs", "getStackLogs")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/updates", "getStackUpdates")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/updates/latest", "getLatestStackUpdate")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/updates/{version}", "getStackUpdate")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/updates/{version}/contents/files", "getUpdateContentsFiles")
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/updates/{version}/contents/file/{path:.*}", "getUpdateContentsFilePath")
|
|
|
|
// The APIs for performing updates of various kind all have the same set of API endpoints. Only
|
|
// differentiate the "create update of kind X" APIs, and introduce a pseudo route param "updateKind".
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/destroy", "createDestroy")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/preview", "createPreview")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/update", "createUpdate")
|
|
|
|
addEndpoint("GET", "/api/stacks/{orgName}/{projectName}/{stackName}/{updateKind}/{updateID}", "getUpdateStatus")
|
|
addEndpoint("PATCH", "/api/stacks/{orgName}/{projectName}/{stackName}/{updateKind}/{updateID}/checkpoint", "patchCheckpoint")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/{updateKind}/{updateID}/complete", "completeUpdate")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/{updateKind}/{updateID}/events", "postEngineEvent")
|
|
addEndpoint("POST", "/api/stacks/{orgName}/{projectName}/{stackName}/{updateKind}/{updateID}/renew_lease", "renewLease")
|
|
|
|
// APIs for managing `PolicyPack`s.
|
|
addEndpoint("POST", "/api/orgs/{orgName}/policypacks", "publishPolicyPack")
|
|
}
|