blob: fdbe40cdd9593c57361b3142fcd354acded06ccb [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 whisk
import (
"errors"
"fmt"
"github.com/apache/openwhisk-client-go/wski18n"
"net/http"
"net/url"
"strconv"
"time"
)
type ActivationService struct {
client *Client
}
type Activation struct {
Namespace string `json:"namespace"`
Name string `json:"name"`
Version string `json:"version"`
Subject string `json:"subject"`
ActivationID string `json:"activationId"`
Cause string `json:"cause,omitempty"`
Start int64 `json:"start"` // When action started (in milliseconds since January 1, 1970 UTC)
End int64 `json:"end"` // Since a 0 is a valid value from server, don't omit
Duration int64 `json:"duration"` // Only available for actions
StatusCode int `json:"statusCode"`
Response `json:"response"`
Logs []string `json:"logs"`
Annotations KeyValueArr `json:"annotations"`
Publish *bool `json:"publish,omitempty"`
}
type ActivationFilteredRow struct {
Row Activation
HeaderFmt string
RowFmt string
}
type Response struct {
Status string `json:"status"`
StatusCode int `json:"statusCode"`
Success bool `json:"success"`
Result *Result `json:"result,omitempty"`
}
type Result map[string]interface{}
type ActivationListOptions struct {
Name string `url:"name,omitempty"`
Limit int `url:"limit"`
Skip int `url:"skip"`
Since int64 `url:"since,omitempty"`
Upto int64 `url:"upto,omitempty"`
Docs bool `url:"docs,omitempty"`
}
//MWD - This structure may no longer be needed as the log format is now a string and not JSON
type Log struct {
Log string `json:"log,omitempty"`
Stream string `json:"stream,omitempty"`
Time string `json:"time,omitempty"`
}
// Status codes to descriptions
var StatusCodes = []string{"success", "application error", "developer error", "internal error"}
// Compare(sortable) compares activation to sortable for the purpose of sorting.
// REQUIRED: sortable must also be of type Activation.
// ***Method of type Sortable***
// ***Currently, no method of sorting defined***
func (activation Activation) Compare(sortable Sortable) bool {
return true
}
// Compare(sortable) compares activation to sortable for the purpose of sorting.
// REQUIRED: sortable must also be of type Activation.
// ***Method of type Sortable***
// ***Currently, no method of sorting defined***
func (activation ActivationFilteredRow) Compare(sortable Sortable) bool {
return true
}
// ToHeaderString() returns the header for a list of activations
func (activation ActivationFilteredRow) ToHeaderString() string {
return fmt.Sprintf(activation.HeaderFmt, "Datetime", "Activation ID", "Kind", "Start", "Duration", "Status", "Entity")
}
// TruncateStr() returns the string, truncated with ...in the middle if it exceeds the specified length
func TruncateStr(str string, maxlen int) string {
if len(str) <= maxlen {
return str
} else {
mid := maxlen / 2
upp := len(str) - mid + 3
if maxlen%2 != 0 {
mid++
}
return str[0:mid] + "..." + str[upp:]
}
}
// ToSummaryRowString() returns a compound string of required parameters for printing
// from CLI command `wsk activation list`.
// ***Method of type Sortable***
func (activation ActivationFilteredRow) ToSummaryRowString() string {
s := time.Unix(0, activation.Row.Start*1000000)
e := time.Unix(0, activation.Row.End*1000000)
var duration = e.Sub(s)
var kind interface{} = activation.Row.Annotations.GetValue("kind")
var initTime interface{} = activation.Row.Annotations.GetValue("initTime")
var status = StatusCodes[0] // assume success
var start = "warm" // assume warm
var fqn = TruncateStr(activation.Row.Namespace, 20) + "/" + TruncateStr(activation.Row.Name, 30) + ":" + TruncateStr(activation.Row.Version, 20)
if activation.Row.Duration == 0 {
duration = s.Sub(s)
}
if kind == nil {
kind = "unknown"
}
if activation.Row.StatusCode > 0 && activation.Row.StatusCode < len(StatusCodes) {
status = StatusCodes[activation.Row.StatusCode]
}
if initTime != nil {
start = "cold"
}
return fmt.Sprintf(
activation.RowFmt+strconv.Itoa(len(fqn))+"s\n",
s.Year(), s.Month(), s.Day(), s.Hour(), s.Minute(), s.Second(),
activation.Row.ActivationID,
kind.(string),
start,
duration,
status,
fqn)
}
func (s *ActivationService) List(options *ActivationListOptions) ([]Activation, *http.Response, error) {
// TODO :: for some reason /activations only works with "_" as namespace
s.client.Namespace = "_"
route := "activations"
routeUrl, err := addRouteOptions(route, options)
if err != nil {
Debug(DbgError, "addRouteOptions(%s, %#v) error: '%s'\n", route, options, err)
errStr := wski18n.T("Unable to append options '{{.options}}' to URL route '{{.route}}': {{.err}}",
map[string]interface{}{"options": fmt.Sprintf("%#v", options), "route": route, "err": err})
werr := MakeWskErrorFromWskError(errors.New(errStr), err, EXIT_CODE_ERR_GENERAL, DISPLAY_MSG, NO_DISPLAY_USAGE)
return nil, nil, werr
}
req, err := s.client.NewRequestUrl("GET", routeUrl, nil, IncludeNamespaceInUrl, AppendOpenWhiskPathPrefix, EncodeBodyAsJson, AuthRequired)
if err != nil {
Debug(DbgError, "http.NewRequestUrl(GET, %s, nil, IncludeNamespaceInUrl, AppendOpenWhiskPathPrefix, EncodeBodyAsJson, AuthRequired) error: '%s'\n", route, err)
errStr := wski18n.T("Unable to create HTTP request for GET '{{.route}}': {{.err}}",
map[string]interface{}{"route": route, "err": err})
werr := MakeWskErrorFromWskError(errors.New(errStr), err, EXIT_CODE_ERR_GENERAL, DISPLAY_MSG, NO_DISPLAY_USAGE)
return nil, nil, werr
}
Debug(DbgInfo, "Sending HTTP request - URL '%s'; req %#v\n", req.URL.String(), req)
var activations []Activation
resp, err := s.client.Do(req, &activations, ExitWithSuccessOnTimeout)
if err != nil {
Debug(DbgError, "s.client.Do() error - HTTP req %s; error '%s'\n", req.URL.String(), err)
return nil, resp, err
}
return activations, resp, nil
}
func (s *ActivationService) Get(activationID string) (*Activation, *http.Response, error) {
// TODO :: for some reason /activations/:id only works with "_" as namespace
s.client.Namespace = "_"
// Encode resource name as a path (with no query params) before inserting it into the URI
// This way any '?' chars in the name won't be treated as the beginning of the query params
activationID = (&url.URL{Path: activationID}).String()
route := fmt.Sprintf("activations/%s", activationID)
req, err := s.client.NewRequest("GET", route, nil, IncludeNamespaceInUrl)
if err != nil {
Debug(DbgError, "http.NewRequest(GET, %s) error: '%s'\n", route, err)
errStr := wski18n.T("Unable to create HTTP request for GET '{{.route}}': {{.err}}",
map[string]interface{}{"route": route, "err": err})
werr := MakeWskErrorFromWskError(errors.New(errStr), err, EXIT_CODE_ERR_GENERAL, DISPLAY_MSG, NO_DISPLAY_USAGE)
return nil, nil, werr
}
Debug(DbgInfo, "Sending HTTP request - URL '%s'; req %#v\n", req.URL.String(), req)
a := new(Activation)
resp, err := s.client.Do(req, &a, ExitWithSuccessOnTimeout)
if err != nil {
Debug(DbgError, "s.client.Do() error - HTTP req %s; error '%s'\n", req.URL.String(), err)
return nil, resp, err
}
a.StatusCode = GetStatusCodeForMessage(a.Status)
return a, resp, nil
}
func GetStatusCodeForMessage(msg string) int {
var code int
switch msg {
case "application error":
code = 1
case "action developer error":
code = 2
case "whisk internal error":
code = 3
default:
code = 0
}
return code
}
func (s *ActivationService) Logs(activationID string) (*Activation, *http.Response, error) {
// TODO :: for some reason /activations/:id/logs only works with "_" as namespace
s.client.Namespace = "_"
// Encode resource name as a path (with no query params) before inserting it into the URI
// This way any '?' chars in the name won't be treated as the beginning of the query params
activationID = (&url.URL{Path: activationID}).String()
route := fmt.Sprintf("activations/%s/logs", activationID)
req, err := s.client.NewRequest("GET", route, nil, IncludeNamespaceInUrl)
if err != nil {
Debug(DbgError, "http.NewRequest(GET, %s) error: '%s'\n", route, err)
errStr := wski18n.T("Unable to create HTTP request for GET '{{.route}}': {{.err}}",
map[string]interface{}{"route": route, "err": err})
werr := MakeWskErrorFromWskError(errors.New(errStr), err, EXIT_CODE_ERR_GENERAL, DISPLAY_MSG, NO_DISPLAY_USAGE)
return nil, nil, werr
}
Debug(DbgInfo, "Sending HTTP request - URL '%s'; req %#v\n", req.URL.String(), req)
activation := new(Activation)
resp, err := s.client.Do(req, &activation, ExitWithSuccessOnTimeout)
if err != nil {
Debug(DbgError, "s.client.Do() error - HTTP req %s; error '%s'\n", req.URL.String(), err)
return nil, resp, err
}
return activation, resp, nil
}
func (s *ActivationService) Result(activationID string) (*Response, *http.Response, error) {
// TODO :: for some reason /activations only works with "_" as namespace
s.client.Namespace = "_"
// Encode resource name as a path (with no query params) before inserting it into the URI
// This way any '?' chars in the name won't be treated as the beginning of the query params
activationID = (&url.URL{Path: activationID}).String()
route := fmt.Sprintf("activations/%s/result", activationID)
req, err := s.client.NewRequest("GET", route, nil, IncludeNamespaceInUrl)
if err != nil {
Debug(DbgError, "http.NewRequest(GET, %s) error: '%s'\n", route, err)
errStr := wski18n.T("Unable to create HTTP request for GET '{{.route}}': {{.err}}",
map[string]interface{}{"route": route, "err": err})
werr := MakeWskErrorFromWskError(errors.New(errStr), err, EXIT_CODE_ERR_GENERAL, DISPLAY_MSG, NO_DISPLAY_USAGE)
return nil, nil, werr
}
Debug(DbgInfo, "Sending HTTP request - URL '%s'; req %#v\n", req.URL.String(), req)
r := new(Response)
resp, err := s.client.Do(req, &r, ExitWithSuccessOnTimeout)
if err != nil {
Debug(DbgError, "s.client.Do() error - HTTP req %s; error '%s'\n", req.URL.String(), err)
return nil, resp, err
}
return r, resp, nil
}