traffic_ops/traffic_ops_golang/status/statuses.go (227 lines of code) (raw):

package status /* * 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. */ import ( "fmt" "net/http" "strconv" "time" "github.com/apache/trafficcontrol/v8/lib/go-tc" "github.com/apache/trafficcontrol/v8/lib/go-tc/tovalidate" "github.com/apache/trafficcontrol/v8/lib/go-util" "github.com/apache/trafficcontrol/v8/traffic_ops/traffic_ops_golang/api" "github.com/apache/trafficcontrol/v8/traffic_ops/traffic_ops_golang/dbhelpers" validation "github.com/go-ozzo/ozzo-validation" ) // we need a type alias to define functions on type TOStatusV5 struct { api.APIInfoImpl `json:"-"` tc.StatusV5 } func (v *TOStatusV5) GetLastUpdated() (*time.Time, bool, error) { return api.GetLastUpdated(v.APIInfo().Tx, *v.ID, "status") } func (v *TOStatusV5) SelectMaxLastUpdatedQuery(where, orderBy, pagination, tableName string) string { return `SELECT max(t) from ( SELECT max(last_updated) as t from ` + tableName + ` s ` + where + orderBy + pagination + ` UNION ALL select max(last_updated) as t from last_deleted l where l.table_name='` + tableName + `') as res` } func (v *TOStatusV5) SetLastUpdated(t tc.TimeNoMod) { v.LastUpdated = &t.Time } func (v *TOStatusV5) InsertQuery() string { return insertQuery() } func (v *TOStatusV5) NewReadObj() interface{} { return &TOStatusV5{} } func (v *TOStatusV5) SelectQuery() string { return selectQuery() } func (v *TOStatusV5) ParamColumns() map[string]dbhelpers.WhereColumnInfo { return map[string]dbhelpers.WhereColumnInfo{ "id": dbhelpers.WhereColumnInfo{Column: "id", Checker: api.IsInt}, "description": dbhelpers.WhereColumnInfo{Column: "description"}, "name": dbhelpers.WhereColumnInfo{Column: "name"}, } } func (v *TOStatusV5) UpdateQuery() string { return updateQuery() } func (v *TOStatusV5) DeleteQuery() string { return deleteQuery() } func (status TOStatusV5) GetKeyFieldsInfo() []api.KeyFieldInfo { return []api.KeyFieldInfo{{Field: "id", Func: api.GetIntKey}} } // Implementation of the Identifier, Validator interface functions func (status TOStatusV5) GetKeys() (map[string]interface{}, bool) { if status.ID == nil { return map[string]interface{}{"id": 0}, false } return map[string]interface{}{"id": *status.ID}, true } func (status *TOStatusV5) SetKeys(keys map[string]interface{}) { i, _ := keys["id"].(int) //this utilizes the non panicking type assertion, if the thrown away ok variable is false i will be the zero of the type, 0 here. status.ID = &i } func (status TOStatusV5) GetAuditName() string { if status.Name != nil { return *status.Name } if status.ID != nil { return strconv.Itoa(*status.ID) } return "unknown" } func (status TOStatusV5) GetType() string { return "status" } func (status TOStatusV5) Validate() (error, error) { errs := validation.Errors{ "name": validation.Validate(status.Name, validation.NotNil, validation.Required), } return util.JoinErrs(tovalidate.ToErrors(errs)), nil } func (st *TOStatusV5) Read(h http.Header, useIMS bool) ([]interface{}, error, error, int, *time.Time) { errCode := http.StatusOK api.DefaultSort(st.APIInfo(), "name") readVals, userErr, sysErr, errCode, maxTime := api.GenericRead(h, st, useIMS) if userErr != nil || sysErr != nil { return nil, userErr, sysErr, errCode, nil } for _, iStatus := range readVals { _, ok := iStatus.(*TOStatusV5) if !ok { return nil, nil, fmt.Errorf("TOStatusV5.Read: api.GenericRead returned unexpected type %T\n", iStatus), http.StatusInternalServerError, nil } } return readVals, nil, nil, errCode, maxTime } func (st *TOStatusV5) Update(h http.Header) (error, error, int) { var statusName string err := st.APIInfo().Tx.QueryRow(`SELECT name from status WHERE id = $1`, *st.ID).Scan(&statusName) if err != nil { return nil, fmt.Errorf("error querying status name from ID: %w", err), http.StatusInternalServerError } if tc.IsReservedStatus(statusName) { return fmt.Errorf("cannot modify %s status", statusName), nil, http.StatusForbidden } return api.GenericUpdate(h, st) } func (st *TOStatusV5) Create() (error, error, int) { return api.GenericCreate(st) } func (st *TOStatusV5) Delete() (error, error, int) { var statusName string err := st.APIInfo().Tx.QueryRow(`SELECT name from status WHERE id = $1`, *st.ID).Scan(&statusName) if err != nil { return nil, fmt.Errorf("error querying status name from ID: %w", err), http.StatusInternalServerError } if tc.IsReservedStatus(statusName) { return fmt.Errorf("cannot delete %s status", statusName), nil, http.StatusForbidden } return api.GenericDelete(st) } // we need a type alias to define functions on type TOStatus struct { api.APIInfoImpl `json:"-"` tc.StatusNullable } func (v *TOStatus) GetLastUpdated() (*time.Time, bool, error) { return api.GetLastUpdated(v.APIInfo().Tx, *v.ID, "status") } func (v *TOStatus) SelectMaxLastUpdatedQuery(where, orderBy, pagination, tableName string) string { return `SELECT max(t) from ( SELECT max(last_updated) as t from ` + tableName + ` s ` + where + orderBy + pagination + ` UNION ALL select max(last_updated) as t from last_deleted l where l.table_name='` + tableName + `') as res` } func (v *TOStatus) SetLastUpdated(t tc.TimeNoMod) { v.LastUpdated = &t } func (v *TOStatus) InsertQuery() string { return insertQuery() } func (v *TOStatus) NewReadObj() interface{} { return &TOStatus{} } func (v *TOStatus) SelectQuery() string { return selectQuery() } func (v *TOStatus) ParamColumns() map[string]dbhelpers.WhereColumnInfo { return map[string]dbhelpers.WhereColumnInfo{ "id": dbhelpers.WhereColumnInfo{Column: "id", Checker: api.IsInt}, "description": dbhelpers.WhereColumnInfo{Column: "description"}, "name": dbhelpers.WhereColumnInfo{Column: "name"}, } } func (v *TOStatus) UpdateQuery() string { return updateQuery() } func (v *TOStatus) DeleteQuery() string { return deleteQuery() } func (status TOStatus) GetKeyFieldsInfo() []api.KeyFieldInfo { return []api.KeyFieldInfo{{Field: "id", Func: api.GetIntKey}} } // Implementation of the Identifier, Validator interface functions func (status TOStatus) GetKeys() (map[string]interface{}, bool) { if status.ID == nil { return map[string]interface{}{"id": 0}, false } return map[string]interface{}{"id": *status.ID}, true } func (status *TOStatus) SetKeys(keys map[string]interface{}) { i, _ := keys["id"].(int) //this utilizes the non panicking type assertion, if the thrown away ok variable is false i will be the zero of the type, 0 here. status.ID = &i } func (status TOStatus) GetAuditName() string { if status.Name != nil { return *status.Name } if status.ID != nil { return strconv.Itoa(*status.ID) } return "unknown" } func (status TOStatus) GetType() string { return "status" } func (status TOStatus) Validate() (error, error) { errs := validation.Errors{ "name": validation.Validate(status.Name, validation.NotNil, validation.Required), } return util.JoinErrs(tovalidate.ToErrors(errs)), nil } func (st *TOStatus) Read(h http.Header, useIMS bool) ([]interface{}, error, error, int, *time.Time) { errCode := http.StatusOK api.DefaultSort(st.APIInfo(), "name") readVals, userErr, sysErr, errCode, maxTime := api.GenericRead(h, st, useIMS) if userErr != nil || sysErr != nil { return nil, userErr, sysErr, errCode, nil } for _, iStatus := range readVals { _, ok := iStatus.(*TOStatus) if !ok { return nil, nil, fmt.Errorf("TOStatus.Read: api.GenericRead returned unexpected type %T\n", iStatus), http.StatusInternalServerError, nil } } return readVals, nil, nil, errCode, maxTime } func (st *TOStatus) Update(h http.Header) (error, error, int) { var statusName string err := st.APIInfo().Tx.QueryRow(`SELECT name from status WHERE id = $1`, *st.ID).Scan(&statusName) if err != nil { return nil, fmt.Errorf("error querying status name from ID: %w", err), http.StatusInternalServerError } if tc.IsReservedStatus(statusName) { return fmt.Errorf("cannot modify %s status", statusName), nil, http.StatusForbidden } return api.GenericUpdate(h, st) } func (st *TOStatus) Create() (error, error, int) { return api.GenericCreate(st) } func (st *TOStatus) Delete() (error, error, int) { var statusName string err := st.APIInfo().Tx.QueryRow(`SELECT name from status WHERE id = $1`, *st.ID).Scan(&statusName) if err != nil { return nil, fmt.Errorf("error querying status name from ID: %w", err), http.StatusInternalServerError } if tc.IsReservedStatus(statusName) { return fmt.Errorf("cannot delete %s status", statusName), nil, http.StatusForbidden } return api.GenericDelete(st) } func selectQuery() string { return ` SELECT description, id, last_updated, name FROM status s ` } func updateQuery() string { return `UPDATE status SET name=:name, description=:description WHERE id=:id RETURNING last_updated` } func insertQuery() string { return `INSERT INTO status ( name, description) VALUES ( :name, :description) RETURNING id,last_updated` } func deleteQuery() string { return `DELETE FROM status WHERE id=:id` }