Github user necouchman commented on a diff in the pull request:
https://github.com/apache/guacamole-client/pull/291#discussion_r191717209
--- Diff:
guacamole/src/main/java/org/apache/guacamole/rest/RESTExceptionMapper.java ---
@@ -0,0 +1,118 @@
+/*
+ * 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.
+ */
+
+package org.apache.guacamole.rest;
+
+import com.google.inject.Inject;
+import com.google.inject.Singleton;
+import javax.servlet.http.HttpServletRequest;
+import javax.ws.rs.core.Context;
+import javax.ws.rs.core.MediaType;
+import javax.ws.rs.core.Response;
+import javax.ws.rs.ext.ExceptionMapper;
+import javax.ws.rs.ext.Provider;
+import org.apache.guacamole.GuacamoleException;
+import org.apache.guacamole.GuacamoleUnauthorizedException;
+import org.apache.guacamole.rest.auth.AuthenticationService;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * A class that maps GuacamoleExceptions in a way that returns a
+ * custom response to the user via JSON rather than allowing the default
+ * web application error handling to take place.
+ */
+@Provider
+@Singleton
+public class RESTExceptionMapper implements ExceptionMapper<Throwable> {
+
+ /**
+ * The logger for this class.
+ */
+ private final Logger logger =
LoggerFactory.getLogger(RESTExceptionMapper.class);
+
+ /**
+ * The request associated with this instance of this mapper.
+ */
+ @Context
+ private HttpServletRequest request;
+
+ /**
+ * The authentication service associated with the currently active
session.
+ */
+ @Inject
+ private AuthenticationService authenticationService;
+
+ /**
+ * Returns the authentication token that is in use in the current
session,
+ * if present, or null if otherwise.
+ *
+ * @return
+ * The authentication token for the current session, or null if no
+ * token is present.
+ */
+ private String getAuthenticationToken() {
+
+ String token = request.getParameter("token");
+ if (token != null && !token.isEmpty())
+ return token;
+
+ return null;
+
+ }
+
+ @Override
+ public Response toResponse(Throwable t) {
+
+ // Ensure any associated session is invalidated if unauthorized
+ if (t instanceof GuacamoleUnauthorizedException) {
+ String token = getAuthenticationToken();
+
+ if (authenticationService.destroyGuacamoleSession(token))
+ logger.debug("Implicitly invalidated session for token
\"{}\"", token);
+ }
+
+ // Translate GuacamoleException subclasses to HTTP error codes
+ if (t instanceof GuacamoleException)
+ return Response
+ .status(((GuacamoleException) t).getHttpStatusCode())
+ .entity(new APIError((GuacamoleException)t))
+ .type(MediaType.APPLICATION_JSON)
+ .build();
+
+ // Rethrow unchecked exceptions such that they are properly
wrapped
+ String message = t.getMessage();
+ if (message != null)
+ logger.error("Unexpected internal error: {}", message);
+ else
+ logger.error("An internal error occurred, but did not contain "
+ + "an error message. Enable debug-level logging for "
+ + "details.");
+
+ logger.debug("Unexpected error in REST endpoint.", t);
+
+ return Response
+ .status(Response.Status.INTERNAL_SERVER_ERROR)
+ .entity("Unexpected Internal Error.")
--- End diff --
Yeah, probably so. Should be good, now.
---