blob: ea49e8ddb83f912dc7bb1cff494d01fc9df8ff9a [file] [log] [blame]
package servercheck
/*
* 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 (
"database/sql"
"errors"
"fmt"
"net/http"
"github.com/apache/trafficcontrol/traffic_ops/traffic_ops_golang/dbhelpers"
"github.com/apache/trafficcontrol/lib/go-tc"
"github.com/apache/trafficcontrol/lib/go-util"
"github.com/apache/trafficcontrol/traffic_ops/traffic_ops_golang/api"
)
const ServerCheck_Get_Endpoint = "GET /servercheck"
const serverInfoQuery = `
SELECT server.host_name AS hostName,
server.id AS id,
profile.name AS profile,
status.name AS adminState,
cachegroup.name AS cacheGroup,
type.name AS type,
server.config_update_time > server.config_apply_time AS upd_pending,
server.revalidate_update_time > server.revalidate_apply_time AS reval_pending
FROM server
LEFT JOIN profile ON server.profile = profile.id
LEFT JOIN status ON server.status = status.id
LEFT JOIN cachegroup ON server.cachegroup = cachegroup.id
LEFT JOIN type ON server.type = type.id
`
const serverChecksQuery = `
SELECT servercheck.id,
servercheck.server,
servercheck.aa,
servercheck.ab,
servercheck.ac,
servercheck.ad,
servercheck.ae,
servercheck.af,
servercheck.ag,
servercheck.ah,
servercheck.ai,
servercheck.aj,
servercheck.ak,
servercheck.al,
servercheck.am,
servercheck.an,
servercheck.ao,
servercheck.ap,
servercheck.aq,
servercheck.ar,
servercheck.bf,
servercheck.at,
servercheck.au,
servercheck.av,
servercheck.aw,
servercheck.ax,
servercheck.ay,
servercheck.az,
servercheck.ba,
servercheck.bb,
servercheck.bc,
servercheck.bd,
servercheck.be
FROM servercheck
`
const extensionsQuery = `
SELECT to_extension.servercheck_column_name,
to_extension.servercheck_short_name
FROM to_extension
LEFT JOIN type ON type.id = to_extension.type
WHERE (type.name = 'CHECK_EXTENSION_BOOL' OR
type.name = 'CHECK_EXTENSION_NUM') AND
to_extension.servercheck_short_name IS NOT NULL AND
to_extension.servercheck_column_name IS NOT NULL
`
// CreateUpdateServercheck handles creating or updating an existing servercheck
func CreateUpdateServercheck(w http.ResponseWriter, r *http.Request) {
inf, userErr, sysErr, errCode := api.NewInfo(r, nil, nil)
if userErr != nil || sysErr != nil {
api.HandleErr(w, r, inf.Tx.Tx, errCode, userErr, sysErr)
return
}
defer inf.Close()
if inf.User.UserName != "extension" {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusForbidden, errors.New("invalid user for this API. Only the \"extension\" user can use this"), nil)
return
}
serverCheckReq := tc.ServercheckRequestNullable{}
if err := api.Parse(r.Body, inf.Tx.Tx, &serverCheckReq); err != nil {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusBadRequest, err, nil)
return
}
id, exists, err := getServerID(serverCheckReq.ID, serverCheckReq.HostName, inf.Tx.Tx)
if err != nil {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusInternalServerError, nil, errors.New("getting server id: "+err.Error()))
return
}
if !exists {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusBadRequest, fmt.Errorf("Server not found"), nil)
return
}
col, exists, err := getColName(serverCheckReq.Name, inf.Tx.Tx)
if err != nil {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusInternalServerError, nil, errors.New("getting servercheck column name: "+err.Error()))
return
}
if !exists {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusBadRequest, fmt.Errorf("Server Check Extension %v not found - Do you need to install it?", *serverCheckReq.Name), nil)
return
}
err = createUpdateServerCheck(id, col, *serverCheckReq.Value, inf.Tx.Tx)
if err != nil {
api.HandleErr(w, r, inf.Tx.Tx, http.StatusInternalServerError, nil, errors.New("updating servercheck: "+err.Error()))
return
}
// NOTE: this endpoint does not create an audit log entry in order to prevent
// spamming the audit log with thousands of entries every minute
api.WriteRespAlert(w, r, tc.SuccessLevel, "Server Check was successfully updated")
}
func getServerID(id *int, hostname *string, tx *sql.Tx) (int, bool, error) {
if id != nil {
_, exists, err := dbhelpers.GetServerNameFromID(tx, int64(*id))
return *id, exists, err
}
sID, exists, err := dbhelpers.GetServerIDFromName(*hostname, tx)
return sID, exists, err
}
func getColName(shortName *string, tx *sql.Tx) (string, bool, error) {
col := ""
if err := tx.QueryRow(`SELECT servercheck_column_name FROM to_extension WHERE servercheck_short_name = $1`, *shortName).Scan(&col); err != nil {
if err == sql.ErrNoRows {
return "", false, nil
}
return "", false, errors.New("querying servercheck column name: " + err.Error())
}
return col, true, nil
}
func createUpdateServerCheck(sid int, colName string, value int, tx *sql.Tx) error {
insertUpdateQuery := fmt.Sprintf(`
INSERT INTO servercheck (server, %[1]s)
VALUES ($1, $2)
ON CONFLICT (server)
DO UPDATE SET %[1]s = EXCLUDED.%[1]s`, colName)
result, err := tx.Exec(insertUpdateQuery, sid, value)
if err != nil {
return errors.New("insert server check: " + err.Error())
}
rowsAffected, err := result.RowsAffected()
if err != nil {
return errors.New("reading rows affected after server check insert : " + err.Error())
}
if rowsAffected == 0 {
return errors.New("server check was inserted, no rows were affected : " + err.Error())
}
return nil
}
// ReadServerCheck is the handler for GET requests for /servercheck
func ReadServerCheck(w http.ResponseWriter, r *http.Request) {
inf, userErr, sysErr, errCode := api.NewInfo(r, nil, nil)
tx := inf.Tx.Tx
if userErr != nil || sysErr != nil {
api.HandleErr(w, r, tx, errCode, userErr, sysErr)
return
}
defer inf.Close()
data, userErr, sysErr, errCode := handleReadServerCheck(inf, tx)
if userErr != nil || sysErr != nil {
api.HandleErr(w, r, tx, errCode, userErr, sysErr)
return
}
api.WriteResp(w, r, data)
}
func handleReadServerCheck(inf *api.APIInfo, tx *sql.Tx) ([]tc.GenericServerCheck, error, error, int) {
extensions := make(map[string]string)
// Query Parameters to Database Query column mappings
queryParamsToQueryCols := map[string]dbhelpers.WhereColumnInfo{
"id": dbhelpers.WhereColumnInfo{Column: "servercheck.server", Checker: api.IsInt},
"hostName": dbhelpers.WhereColumnInfo{Column: "server.host_name"},
}
where, orderBy, pagination, queryValues, errs := dbhelpers.BuildWhereAndOrderByAndPagination(inf.Params, queryParamsToQueryCols)
if len(errs) > 0 {
return nil, util.JoinErrs(errs), nil, http.StatusBadRequest
}
// where clause is different for servercheck and server table. Also, it differs for the query param.
var whereSC, whereSI string
if len(inf.Params) < 1 {
whereSI = "WHERE type.name LIKE 'MID%' OR type.name LIKE 'EDGE%' "
whereSC = ""
} else if len(inf.Params) == 1 {
if _, ok := inf.Params["hostName"]; ok {
whereSI = "WHERE (type.name LIKE 'MID%' OR type.name LIKE 'EDGE%') AND server.host_name=:hostName "
whereSC = ""
} else if _, ok = inf.Params["id"]; ok {
whereSI = "WHERE (type.name LIKE 'MID%' OR type.name LIKE 'EDGE%') AND server.id=:id "
whereSC = where
} else {
whereSI = "WHERE type.name LIKE 'MID%' OR type.name LIKE 'EDGE%' "
whereSC = ""
}
} else if len(inf.Params) > 1 {
_, ok := inf.Params["id"]
_, ok1 := inf.Params["hostName"]
if ok && ok1 {
whereSI = "WHERE (type.name LIKE 'MID%' OR type.name LIKE 'EDGE%') AND (server.host_name=:hostName AND server.id=:id)"
whereSC = "WHERE servercheck.server=:id"
} else if ok && !ok1 {
whereSI = "WHERE (type.name LIKE 'MID%' OR type.name LIKE 'EDGE%') AND server.id=:id "
whereSC = "WHERE servercheck.server=:id"
} else if ok1 && !ok {
whereSI = "WHERE (type.name LIKE 'MID%' OR type.name LIKE 'EDGE%') AND server.host_name=:hostName "
whereSC = ""
} else {
whereSI = "WHERE type.name LIKE 'MID%' OR type.name LIKE 'EDGE%' "
whereSC = ""
}
}
extRows, err := tx.Query(extensionsQuery)
if err != nil {
sysErr := fmt.Errorf("querying for extensions: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
for extRows.Next() {
var shortName string
var checkName string
if err = extRows.Scan(&shortName, &checkName); err != nil {
sysErr := fmt.Errorf("scanning extension: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
extensions[shortName] = checkName
}
querySC := serverChecksQuery + whereSC + orderBy + pagination
colRows, err := inf.Tx.NamedQuery(querySC, queryValues)
if err != nil {
sysErr := fmt.Errorf("querying serverchecks columns: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
columns := make(map[int]tc.ServerCheckColumns)
for colRows.Next() {
var cols tc.ServerCheckColumns
if err = colRows.StructScan(&cols); err != nil {
sysErr := fmt.Errorf("scanning server checks columns: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
columns[cols.Server] = cols
}
orderBySI := orderBy + "ORDER BY hostName ASC"
querySI := serverInfoQuery + whereSI + orderBySI + pagination
serverRows, err := inf.Tx.NamedQuery(querySI, queryValues)
if err != nil {
sysErr := fmt.Errorf("querying server info for checks: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
data := []tc.GenericServerCheck{}
for serverRows.Next() {
var serverInfo tc.GenericServerCheck
if err = serverRows.Scan(&serverInfo.HostName, &serverInfo.ID, &serverInfo.Profile, &serverInfo.AdminState, &serverInfo.CacheGroup, &serverInfo.Type, &serverInfo.UpdPending, &serverInfo.RevalPending); err != nil {
sysErr := fmt.Errorf("scanning server info for checks: %v", err)
return nil, nil, sysErr, http.StatusInternalServerError
}
serverCheckCols, ok := columns[serverInfo.ID]
if ok {
serverInfo.Checks = make(map[string]*int)
} else {
data = append(data, serverInfo)
continue
}
for colName, checkName := range extensions {
switch colName {
case "aa":
serverInfo.Checks[checkName] = serverCheckCols.AA
case "ab":
serverInfo.Checks[checkName] = serverCheckCols.AB
case "ac":
serverInfo.Checks[checkName] = serverCheckCols.AC
case "ad":
serverInfo.Checks[checkName] = serverCheckCols.AD
case "ae":
serverInfo.Checks[checkName] = serverCheckCols.AE
case "af":
serverInfo.Checks[checkName] = serverCheckCols.AF
case "ag":
serverInfo.Checks[checkName] = serverCheckCols.AG
case "ah":
serverInfo.Checks[checkName] = serverCheckCols.AH
case "ai":
serverInfo.Checks[checkName] = serverCheckCols.AI
case "aj":
serverInfo.Checks[checkName] = serverCheckCols.AJ
case "ak":
serverInfo.Checks[checkName] = serverCheckCols.AK
case "al":
serverInfo.Checks[checkName] = serverCheckCols.AL
case "am":
serverInfo.Checks[checkName] = serverCheckCols.AM
case "an":
serverInfo.Checks[checkName] = serverCheckCols.AN
case "ao":
serverInfo.Checks[checkName] = serverCheckCols.AO
case "ap":
serverInfo.Checks[checkName] = serverCheckCols.AP
case "aq":
serverInfo.Checks[checkName] = serverCheckCols.AQ
case "ar":
serverInfo.Checks[checkName] = serverCheckCols.AR
case "at":
serverInfo.Checks[checkName] = serverCheckCols.AT
case "au":
serverInfo.Checks[checkName] = serverCheckCols.AU
case "av":
serverInfo.Checks[checkName] = serverCheckCols.AV
case "aw":
serverInfo.Checks[checkName] = serverCheckCols.AW
case "ax":
serverInfo.Checks[checkName] = serverCheckCols.AX
case "ay":
serverInfo.Checks[checkName] = serverCheckCols.AY
case "az":
serverInfo.Checks[checkName] = serverCheckCols.AZ
case "ba":
serverInfo.Checks[checkName] = serverCheckCols.BA
case "bb":
serverInfo.Checks[checkName] = serverCheckCols.BB
case "bc":
serverInfo.Checks[checkName] = serverCheckCols.BC
case "bd":
serverInfo.Checks[checkName] = serverCheckCols.BD
case "be":
serverInfo.Checks[checkName] = serverCheckCols.BE
case "bf":
serverInfo.Checks[checkName] = serverCheckCols.BF
}
}
data = append(data, serverInfo)
}
return data, nil, nil, http.StatusOK
}