mirror of
https://github.com/portainer/portainer.git
synced 2025-07-23 07:19:41 +02:00
* refactor(tags): replace tags with tag ids * refactor(tags): revert tags to be strings and add tagids * refactor(tags): enable search by tag in home view * refactor(tags): show endpoint tags * refactor(endpoints): expect tagIds on create payload * refactor(endpoints): expect tagIds on update payload * refactor(endpoints): replace TagIds to TagIDs * refactor(endpoints): set endpoint group to get TagIDs * refactor(endpoints): refactor tag-selector to receive tag-ids * refactor(endpoints): show tags in multi-endpoint-selector * chore(tags): revert reformat * refactor(endpoints): remove unneeded bind * refactor(endpoints): change param tags to tagids in endpoint create * refactor(endpoints): remove console.log * refactor(tags): remove deleted tag from endpoint and endpoint group * fix(endpoints): show loading label while loading tags * chore(go): remove obsolete import labels * chore(db): add db version comment * fix(db): add tag service to migrator * refactor(db): add error checks in migrator * style(db): sort props in alphabetical order * style(tags): fix typo Co-Authored-By: Anthony Lapenna <anthony.lapenna@portainer.io> * refactor(endpoints): replace tagsMap with tag string representation * refactor(tags): rewrite tag delete to be more readable * refactor(home): rearange code to match former style * refactor(tags): guard against missing model in tag-selector * refactor(tags): rename vars in tag_delete * refactor(tags): allow any authenticated user to fetch tag list * refactor(endpoints): replace controller function with class * refactor(endpoints): replace function with helper * refactor(endpoints): replace controller with class * refactor(tags): revert tags-selector to use 1 way bindings * refactor(endpoints): load empty tag array instead of nil * refactor(endpoints): revert default tag ids * refactor(endpoints): use function in place * refactor(tags): use lodash * style(tags): use parens in arrow functions * fix(tags): remove tag from tag model Co-authored-by: Anthony Lapenna <anthony.lapenna@portainer.io>
189 lines
5.6 KiB
Go
189 lines
5.6 KiB
Go
package endpoints
|
|
|
|
import (
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/portainer/portainer/api"
|
|
|
|
"github.com/portainer/libhttp/request"
|
|
|
|
httperror "github.com/portainer/libhttp/error"
|
|
"github.com/portainer/libhttp/response"
|
|
"github.com/portainer/portainer/api/http/security"
|
|
)
|
|
|
|
// GET request on /api/endpoints?(start=<start>)&(limit=<limit>)&(search=<search>)&(groupId=<groupId)
|
|
func (handler *Handler) endpointList(w http.ResponseWriter, r *http.Request) *httperror.HandlerError {
|
|
start, _ := request.RetrieveNumericQueryParameter(r, "start", true)
|
|
if start != 0 {
|
|
start--
|
|
}
|
|
|
|
search, _ := request.RetrieveQueryParameter(r, "search", true)
|
|
if search != "" {
|
|
search = strings.ToLower(search)
|
|
}
|
|
|
|
groupID, _ := request.RetrieveNumericQueryParameter(r, "groupId", true)
|
|
limit, _ := request.RetrieveNumericQueryParameter(r, "limit", true)
|
|
endpointType, _ := request.RetrieveNumericQueryParameter(r, "type", true)
|
|
|
|
endpointGroups, err := handler.EndpointGroupService.EndpointGroups()
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to retrieve endpoint groups from the database", err}
|
|
}
|
|
|
|
endpoints, err := handler.EndpointService.Endpoints()
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to retrieve endpoints from the database", err}
|
|
}
|
|
|
|
securityContext, err := security.RetrieveRestrictedRequestContext(r)
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to retrieve info from request context", err}
|
|
}
|
|
|
|
filteredEndpoints := security.FilterEndpoints(endpoints, endpointGroups, securityContext)
|
|
|
|
if groupID != 0 {
|
|
filteredEndpoints = filterEndpointsByGroupID(filteredEndpoints, portainer.EndpointGroupID(groupID))
|
|
}
|
|
|
|
if search != "" {
|
|
tags, err := handler.TagsService.Tags()
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to retrieve tags from the database", err}
|
|
}
|
|
tagsMap := make(map[portainer.TagID]string)
|
|
for _, tag := range tags {
|
|
tagsMap[tag.ID] = tag.Name
|
|
}
|
|
filteredEndpoints = filterEndpointsBySearchCriteria(filteredEndpoints, endpointGroups, tagsMap, search)
|
|
}
|
|
|
|
if endpointType != 0 {
|
|
filteredEndpoints = filterEndpointsByType(filteredEndpoints, portainer.EndpointType(endpointType))
|
|
}
|
|
|
|
filteredEndpointCount := len(filteredEndpoints)
|
|
|
|
paginatedEndpoints := paginateEndpoints(filteredEndpoints, start, limit)
|
|
|
|
for idx := range paginatedEndpoints {
|
|
hideFields(&paginatedEndpoints[idx])
|
|
}
|
|
|
|
w.Header().Set("X-Total-Count", strconv.Itoa(filteredEndpointCount))
|
|
return response.JSON(w, paginatedEndpoints)
|
|
}
|
|
|
|
func paginateEndpoints(endpoints []portainer.Endpoint, start, limit int) []portainer.Endpoint {
|
|
if limit == 0 {
|
|
return endpoints
|
|
}
|
|
|
|
endpointCount := len(endpoints)
|
|
|
|
if start > endpointCount {
|
|
start = endpointCount
|
|
}
|
|
|
|
end := start + limit
|
|
if end > endpointCount {
|
|
end = endpointCount
|
|
}
|
|
|
|
return endpoints[start:end]
|
|
}
|
|
|
|
func filterEndpointsByGroupID(endpoints []portainer.Endpoint, endpointGroupID portainer.EndpointGroupID) []portainer.Endpoint {
|
|
filteredEndpoints := make([]portainer.Endpoint, 0)
|
|
|
|
for _, endpoint := range endpoints {
|
|
if endpoint.GroupID == endpointGroupID {
|
|
filteredEndpoints = append(filteredEndpoints, endpoint)
|
|
}
|
|
}
|
|
|
|
return filteredEndpoints
|
|
}
|
|
|
|
func filterEndpointsBySearchCriteria(endpoints []portainer.Endpoint, endpointGroups []portainer.EndpointGroup, tagsMap map[portainer.TagID]string, searchCriteria string) []portainer.Endpoint {
|
|
filteredEndpoints := make([]portainer.Endpoint, 0)
|
|
|
|
for _, endpoint := range endpoints {
|
|
endpointTags := convertTagIDsToTags(tagsMap, endpoint.TagIDs)
|
|
if endpointMatchSearchCriteria(&endpoint, endpointTags, searchCriteria) {
|
|
filteredEndpoints = append(filteredEndpoints, endpoint)
|
|
continue
|
|
}
|
|
|
|
if endpointGroupMatchSearchCriteria(&endpoint, endpointGroups, tagsMap, searchCriteria) {
|
|
filteredEndpoints = append(filteredEndpoints, endpoint)
|
|
}
|
|
}
|
|
|
|
return filteredEndpoints
|
|
}
|
|
|
|
func endpointMatchSearchCriteria(endpoint *portainer.Endpoint, tags []string, searchCriteria string) bool {
|
|
if strings.Contains(strings.ToLower(endpoint.Name), searchCriteria) {
|
|
return true
|
|
}
|
|
|
|
if strings.Contains(strings.ToLower(endpoint.URL), searchCriteria) {
|
|
return true
|
|
}
|
|
|
|
if endpoint.Status == portainer.EndpointStatusUp && searchCriteria == "up" {
|
|
return true
|
|
} else if endpoint.Status == portainer.EndpointStatusDown && searchCriteria == "down" {
|
|
return true
|
|
}
|
|
for _, tag := range tags {
|
|
if strings.Contains(strings.ToLower(tag), searchCriteria) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func endpointGroupMatchSearchCriteria(endpoint *portainer.Endpoint, endpointGroups []portainer.EndpointGroup, tagsMap map[portainer.TagID]string, searchCriteria string) bool {
|
|
for _, group := range endpointGroups {
|
|
if group.ID == endpoint.GroupID {
|
|
if strings.Contains(strings.ToLower(group.Name), searchCriteria) {
|
|
return true
|
|
}
|
|
tags := convertTagIDsToTags(tagsMap, group.TagIDs)
|
|
for _, tag := range tags {
|
|
if strings.Contains(strings.ToLower(tag), searchCriteria) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func filterEndpointsByType(endpoints []portainer.Endpoint, endpointType portainer.EndpointType) []portainer.Endpoint {
|
|
filteredEndpoints := make([]portainer.Endpoint, 0)
|
|
|
|
for _, endpoint := range endpoints {
|
|
if endpoint.Type == endpointType {
|
|
filteredEndpoints = append(filteredEndpoints, endpoint)
|
|
}
|
|
}
|
|
return filteredEndpoints
|
|
}
|
|
|
|
func convertTagIDsToTags(tagsMap map[portainer.TagID]string, tagIDs []portainer.TagID) []string {
|
|
tags := make([]string, 0)
|
|
for _, tagID := range tagIDs {
|
|
tags = append(tags, tagsMap[tagID])
|
|
}
|
|
return tags
|
|
}
|