All Downloads are FREE. Search and download functionalities are using the official Maven repository.

sirius.web.services.ServiceDispatcher Maven / Gradle / Ivy

/*
 * Made with all the love in the world
 * by scireum in Remshalden, Germany
 *
 * Copyright by scireum GmbH
 * http://www.scireum.de - [email protected]
 */

package sirius.web.services;

import io.netty.handler.codec.http.HttpResponseStatus;
import sirius.kernel.async.CallContext;
import sirius.kernel.async.TaskContext;
import sirius.kernel.async.Tasks;
import sirius.kernel.commons.PriorityCollector;
import sirius.kernel.commons.Strings;
import sirius.kernel.commons.Tuple;
import sirius.kernel.di.GlobalContext;
import sirius.kernel.di.std.Context;
import sirius.kernel.di.std.Part;
import sirius.kernel.di.std.Register;
import sirius.kernel.nls.NLS;
import sirius.web.http.WebContext;
import sirius.web.http.WebDispatcher;
import sirius.web.security.Permissions;
import sirius.web.security.UserContext;
import sirius.web.security.UserInfo;

/**
 * Dispatches calls to the JSON / XML Service-Framework (/service).
 * 

* Processes calls to /service/[format]/service-name, by dispatching them to the appropriate * {@link StructuredService} and selecting the matching {@link sirius.kernel.xml.StructuredOutput} based on the given * format (either json or xml). */ @Register public class ServiceDispatcher implements WebDispatcher { private static final String SYSTEM_SERVICE = "SERVICE"; @Override public int getPriority() { return PriorityCollector.DEFAULT_PRIORITY - 5; } @Context private GlobalContext gc; @Part private Tasks tasks; @Override public boolean preDispatch(WebContext ctx) throws Exception { return false; } @Override public boolean dispatch(final WebContext ctx) throws Exception { if (!ctx.getRequest().getUri().startsWith("/service")) { return false; } // The real dispatching is put into its own method to support inlining of this check by the JIT return doDispatch(ctx); } /* * Actually tries to dispatch the incoming request which starts with /service.... */ private boolean doDispatch(WebContext ctx) { String uri = ctx.getRequestedURI(); Tuple handler = parsePath(ctx, uri); if (handler.getSecond() == null) { return false; } tasks.executor("web-services") .dropOnOverload(() -> ctx.respondWith() .error(HttpResponseStatus.INTERNAL_SERVER_ERROR, "Request dropped - System overload!")) .fork(() -> invokeService(ctx, handler.getFirst(), handler.getSecond())); return true; } private Tuple parsePath(WebContext ctx, String uri) { // Cut /service/ final String subpath = uri.substring(9); Tuple callPath = Strings.split(subpath, "/"); String type = callPath.getFirst(); String service = callPath.getSecond(); ServiceCall call = null; if ("xml".equals(type)) { call = new XMLServiceCall(ctx); } else if ("json".equals(type)) { call = new JSONServiceCall(ctx); } else { if (Strings.isFilled(service)) { service = type + "/" + service; } else { service = type; } call = new RawServiceCall(ctx); } StructuredService serv = gc.getPart(service, StructuredService.class); return Tuple.create(call, serv); } private void invokeService(WebContext ctx, ServiceCall call, StructuredService serv) { TaskContext.get().setSystem(SYSTEM_SERVICE).setSubSystem(serv.getClass().getSimpleName()); // Install language CallContext.getCurrent().setLang(NLS.makeLang(ctx.getLang())); // Install user. This is forcefully called here to ensure that the ScopeDetetor // and the user manager are guaranteed to be invoked one we enter the service code... UserInfo user = UserContext.getCurrentUser(); // If the underlying ScopeDetector made a redirect (for whatever reasons) // the response will be committed and we can (must) safely return... if (ctx.isResponseCommitted()) { return; } // ... and check permissions for (String p : Permissions.computePermissionsFromAnnotations(serv.getClass())) { if (!user.hasPermission(p)) { ctx.respondWith().error(HttpResponseStatus.UNAUTHORIZED, "Missing permission: " + p); return; } } ctx.enableTiming(null); call.invoke(serv); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy