rest-api/src/jetbrains/buildServer/server/rest/request/AgentRequest.java (260 lines of code) (raw):

/* * Copyright 2000-2024 JetBrains s.r.o. * * 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 jetbrains.buildServer.server.rest.request; import com.intellij.openapi.util.text.StringUtil; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import io.swagger.annotations.ApiParam; import java.util.Date; import javax.servlet.http.HttpServletRequest; import javax.ws.rs.*; import javax.ws.rs.core.Context; import javax.ws.rs.core.UriInfo; import jetbrains.buildServer.ServiceLocator; import jetbrains.buildServer.server.rest.ApiUrlBuilder; import jetbrains.buildServer.server.rest.data.DataProvider; import jetbrains.buildServer.server.rest.data.Locator; import jetbrains.buildServer.server.rest.data.PagedSearchResult; import jetbrains.buildServer.server.rest.data.finder.impl.AgentFinder; import jetbrains.buildServer.server.rest.data.finder.impl.AgentPoolFinder; import jetbrains.buildServer.server.rest.data.finder.impl.UserFinder; import jetbrains.buildServer.server.rest.errors.AuthorizationFailedException; import jetbrains.buildServer.server.rest.errors.BadRequestException; import jetbrains.buildServer.server.rest.model.Fields; import jetbrains.buildServer.server.rest.model.PagerData; import jetbrains.buildServer.server.rest.model.PagerDataImpl; import jetbrains.buildServer.server.rest.model.agent.*; import jetbrains.buildServer.server.rest.model.buildType.BuildTypes; import jetbrains.buildServer.server.rest.swagger.constants.LocatorName; import jetbrains.buildServer.server.rest.util.BeanContext; import jetbrains.buildServer.serverSide.BuildAgentManager; import jetbrains.buildServer.serverSide.SBuildAgent; import jetbrains.buildServer.serverSide.auth.AuthUtil; import jetbrains.buildServer.serverSide.auth.SecurityContext; import jetbrains.buildServer.users.SUser; import org.jetbrains.annotations.NotNull; /** * @author Yegor.Yarko * Date: 01.08.2009 */ @Path(AgentRequest.API_AGENTS_URL) @Api("Agent") public class AgentRequest { @Context private DataProvider myDataProvider; @Context private ApiUrlBuilder myApiUrlBuilder; @Context @NotNull private AgentPoolFinder myAgentPoolFinder; @Context @NotNull private AgentFinder myAgentFinder; @Context @NotNull private ServiceLocator myServiceLocator; @Context @NotNull private BeanContext myBeanContext; public static final String API_AGENTS_URL = Constants.API_URL + "/agents"; public static String getHref() { return API_AGENTS_URL; } public static String getItemsHref(final String locatorText) { return API_AGENTS_URL + "?locator=" + locatorText; } public static String getAgentHref(@NotNull final SBuildAgent agent) { return API_AGENTS_URL + "/" + AgentFinder.getLocator(agent); } /** * Returns list of agents * * @param includeDisconnected Deprecated, use "locator" parameter instead * @param includeUnauthorized Deprecated, use "locator" parameter instead * @param locator * @return */ @GET @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get all known agents.",nickname="getAllAgents") public Agents serveAgents(@ApiParam(hidden = true) @QueryParam("includeDisconnected") Boolean includeDisconnected, @ApiParam(hidden = true) @QueryParam("includeUnauthorized") Boolean includeUnauthorized, @ApiParam(format = LocatorName.AGENT) @QueryParam("locator") String locator, @QueryParam("fields") String fields, @Context UriInfo uriInfo, @Context HttpServletRequest request) { if (locator != null && includeDisconnected != null) { throw new BadRequestException("Both 'includeDisconnected' URL parameter and '" + AgentFinder.CONNECTED + "' locator dimension are specified. Please use locator only."); } if (locator != null && includeUnauthorized != null) { throw new BadRequestException("Both 'includeUnauthorized' URL parameter and '" + AgentFinder.AUTHORIZED + "' locator dimension are specified. Please use locator only."); } String locatorToUse = locator; if (includeDisconnected != null) { //pre-8.1 compatibility: locatorToUse = Locator.createEmptyLocator().setDimensionIfNotPresent(AgentFinder.CONNECTED, String.valueOf(!includeDisconnected)).getStringRepresentation(); } final Locator parsedLocator = StringUtil.isEmpty(locatorToUse) ? Locator.createEmptyLocator() : new Locator(locatorToUse); if (includeUnauthorized != null) { //pre-8.1 compatibility: locatorToUse = parsedLocator.setDimensionIfNotPresent(AgentFinder.AUTHORIZED, String.valueOf(!includeUnauthorized)).getStringRepresentation(); } final PagedSearchResult<SBuildAgent> result = myAgentFinder.getItems(locatorToUse); final PagerData pager = new PagerDataImpl(uriInfo.getRequestUriBuilder(), request.getContextPath(), result, locatorToUse, "locator"); return new Agents(result.getEntries(), pager, new Fields(fields), myBeanContext); } @GET @Path("/{agentLocator}") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get agent matching the locator.",nickname="getAgent") public Agent serveAgent(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { return new Agent(myAgentFinder.getItem(agentLocator), new Fields(fields), myBeanContext); } @DELETE @Path("/{agentLocator}") @ApiOperation(value="Delete an inactive agent.",nickname="deleteAgent") public void deleteAgent(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); myServiceLocator.getSingletonService(BuildAgentManager.class).removeAgent(agent, myServiceLocator.getSingletonService(UserFinder.class).getCurrentUser()); } @GET @Path("/{agentLocator}/pool") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get the agent pool of the matching agent.",nickname="getAgentPool") public AgentPool getAgentPool(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); final jetbrains.buildServer.serverSide.agentPools.AgentPool agentPool = myAgentPoolFinder.getAgentPool(agent); return agentPool == null ? null : new AgentPool(agentPool, new Fields(fields), myBeanContext); } @PUT @Path("/{agentLocator}/pool") @Consumes({"application/xml", "application/json"}) @Produces({"application/xml", "application/json"}) @ApiOperation(value="Assign the matching agent to the specified agent pool.",nickname="setAgentPool") public AgentPool setAgentPool(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, AgentPool agentPool, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); myDataProvider.addAgentToPool(agentPool.getAgentPoolFromPosted(myAgentPoolFinder), agent); final jetbrains.buildServer.serverSide.agentPools.AgentPool foundPool = myAgentPoolFinder.getAgentPool(agent); return new AgentPool(foundPool, new Fields(fields), myBeanContext); } @GET @Path("/{agentLocator}/enabledInfo") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Check if the matching agent is enabled.",nickname="getEnabledInfo") public AgentEnabledInfo getEnabledInfo(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); return new AgentEnabledInfo(agent, new Fields(fields), myBeanContext); } @PUT @Path("/{agentLocator}/enabledInfo") @Consumes({"application/xml", "application/json"}) @Produces({"application/xml", "application/json"}) @ApiOperation(value="Update the enablement status of the matching agent.",nickname="setEnabledInfo") public AgentEnabledInfo setEnabledInfo(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, AgentEnabledInfo enabledInfo, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); if (enabledInfo == null) throw new BadRequestException("No data is sent as payload."); String commentText = enabledInfo.getCommentTextFromPosted(); Boolean value = enabledInfo.getStatusFromPosted(); if (value == null && commentText == null) throw new BadRequestException("Neither value nor comment are provided, nothing to change"); Date switchTime = enabledInfo.getStatusSwitchTimeFromPosted(myServiceLocator); SUser currentUser = myServiceLocator.getSingletonService(UserFinder.class).getCurrentUser(); if (switchTime == null) { agent.setEnabled(value != null ? value : agent.isEnabled(), currentUser, Agent.getActualActionComment(commentText)); } else { agent.setEnabled(value != null ? value : agent.isEnabled(), currentUser, Agent.getActualActionComment(commentText), switchTime.getTime()); } return new AgentEnabledInfo(agent, new Fields(fields), myBeanContext); } @GET @Path("/{agentLocator}/authorizedInfo") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get the authorization info of the matching agent.",nickname="getAuthorizedInfo") public AgentAuthorizedInfo getAuthorizedInfo(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); return new AgentAuthorizedInfo(agent, new Fields(fields), myBeanContext); } @PUT @Path("/{agentLocator}/authorizedInfo") @Consumes({"application/xml", "application/json"}) @Produces({"application/xml", "application/json"}) @ApiOperation(value="Update the authorization info of the matching agent.",nickname="setAuthorizedInfo") public AgentAuthorizedInfo setAuthorizedInfo(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, AgentAuthorizedInfo authorizedInfo, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); if (authorizedInfo == null) throw new BadRequestException("No data is sent as payload."); String commentText = authorizedInfo.getCommentTextFromPosted(); Boolean value = authorizedInfo.getStatusFromPosted(); if (value == null && commentText == null) throw new BadRequestException("Neither value nor comment are provided, nothing to change"); agent.setAuthorized(value != null ? value : agent.isAuthorized(), myServiceLocator.getSingletonService(UserFinder.class).getCurrentUser(), Agent.getActualActionComment(commentText)); return new AgentAuthorizedInfo(agent, new Fields(fields), myBeanContext); } @GET @Path("/{agentLocator}/{field}") @Produces("text/plain") @ApiOperation(value="Get a field of the matching agent.",nickname="getAgentField") public String serveAgentField(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @PathParam("field") String fieldName) { return Agent.getFieldValue(myAgentFinder.getItem(agentLocator), fieldName, myServiceLocator); } /** * Experimental support to get currently compatible build types */ @GET @Path("/{agentLocator}/compatibleBuildTypes") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get build types compatible with the matching agent.",nickname="getCompatibleBuildTypes") public BuildTypes getCompatibleBuildTypes(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); if (!AuthUtil.canViewAgentDetails(myBeanContext.getServiceLocator().getSingletonService(SecurityContext.class).getAuthorityHolder(), agent)) { throw new AuthorizationFailedException("No permission to view agent details"); } Fields fieldsDefinition = new Fields(Agent.COMPATIBLE_BUILD_TYPES + "(" + (StringUtil.isEmpty(fields) ? "$long" : fields) + ")"); return new Agent(agent, fieldsDefinition, myBeanContext).compatibleBuildTypes; } /** * Experimental support to get currently incompatible build types with incompatibility reason */ @GET @Path("/{agentLocator}/incompatibleBuildTypes") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get build types incompatible with the matching agent.",nickname="getIncompatibleBuildTypes") public Compatibilities geIncompatibleBuildTypes(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); if (!AuthUtil.canViewAgentDetails(myBeanContext.getServiceLocator().getSingletonService(SecurityContext.class).getAuthorityHolder(), agent)) { throw new AuthorizationFailedException("No permission to view agent details"); } Fields fieldsDefinition = new Fields(Agent.INCOMPATIBLE_BUILD_TYPES + "(" + (StringUtil.isEmpty(fields) ? "$long" : fields) + ")"); return new Agent(agent, fieldsDefinition, myBeanContext).incompatibleBuildTypes; } /** * Experimental use only */ @GET @Path("/{agentLocator}/compatibilityPolicy") @Produces({"application/xml", "application/json"}) @ApiOperation(value="Get the build configuration run policy of the matching agent.",nickname="getBuildConfigurationRunPolicy") public CompatibilityPolicy getAllowedRunConfigurations(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); return CompatibilityPolicy.getCompatibilityPolicy(agent, new Fields(fields), myBeanContext); } /** * Experimental use only */ @PUT @Path("/{agentLocator}/compatibilityPolicy") @Consumes({"application/xml", "application/json"}) @Produces({"application/xml", "application/json"}) @ApiOperation(value="Update build configuration run policy of agent matching locator.",nickname="setBuildConfigurationRunPolicy") public CompatibilityPolicy setAllowedRunConfigurations(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, CompatibilityPolicy payload, @QueryParam("fields") String fields) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); payload.applyTo(agent, myServiceLocator); return CompatibilityPolicy.getCompatibilityPolicy(agent, new Fields(fields), myBeanContext); } @PUT @Path("/{agentLocator}/{field}") @Consumes("text/plain") @Produces("text/plain") @ApiOperation(value="Update a field of the matching agent.",nickname="setAgentField") public String setAgentField(@ApiParam(format = LocatorName.AGENT) @PathParam("agentLocator") String agentLocator, @PathParam("field") String fieldName, String value) { final SBuildAgent agent = myAgentFinder.getItem(agentLocator); Agent.setFieldValue(agent, fieldName, value, myServiceLocator); return Agent.getFieldValue(agent, fieldName, myServiceLocator); } public static AgentRequest createForTests(@NotNull final BeanContext beanContext) { AgentRequest result = new AgentRequest(); result.myBeanContext = beanContext; result.myServiceLocator = beanContext.getServiceLocator(); result.myAgentFinder = beanContext.getSingletonService(AgentFinder.class); result.myAgentPoolFinder = beanContext.getSingletonService(AgentPoolFinder.class); return result; } }