com.amazon.ask.servlet.SkillServlet Maven / Gradle / Ivy
Show all versions of ask-sdk-servlet-support Show documentation
/*
Copyright 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file
except in compliance with the License. A copy of the License is located at
http://aws.amazon.com/apache2.0/
or in the "license" file accompanying this file. This file 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 com.amazon.ask.servlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.io.IOException;
import java.io.NotSerializableException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.net.Proxy;
import java.util.ArrayList;
import java.util.List;
import com.amazon.ask.Skill;
import com.amazon.ask.exception.AskSdkException;
import com.amazon.ask.model.RequestEnvelope;
import com.amazon.ask.model.services.Serializer;
import com.amazon.ask.request.impl.BaseSkillRequest;
import com.amazon.ask.response.SkillResponse;
import com.amazon.ask.servlet.util.ServletUtils;
import com.amazon.ask.servlet.verifiers.AlexaHttpRequest;
import com.amazon.ask.servlet.verifiers.ServletRequest;
import com.amazon.ask.servlet.verifiers.SkillRequestSignatureVerifier;
import com.amazon.ask.servlet.verifiers.SkillRequestTimestampVerifier;
import com.amazon.ask.servlet.verifiers.SkillServletVerifier;
import com.amazon.ask.util.JacksonSerializer;
import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static com.amazon.ask.servlet.ServletConstants.DEFAULT_TOLERANCE_MILLIS;
/**
*
* Simple implementation of a skill in the form of a Java EE servlet. Use this class when coding the
* skill as a web service.
*
*
* This class takes care of the JSON serialization / deserialization of the HTTP body and the
* invocation of the right method of the provided {@code Skill} . It also handles sending back
* modified session attributes, user attributes and authentication tokens when needed and handles
* exception cases.
*
*/
public class SkillServlet extends HttpServlet {
/**
* Logger mechanism to log data for debugging purposes.
*/
private static final Logger LOGGER = LoggerFactory.getLogger(SkillServlet.class);
/**
* The serialization runtime associates with each serializable class a version number, called a serialVersionUID,
* which is used during deserialization to verify that the sender and receiver of a serialized object have loaded
* classes for that object that are compatible with respect to serialization.
*/
private static final long serialVersionUID = 3257254794185762002L;
/**
* Instance of type {@link Skill}.
*/
private transient final Skill skill;
/**
* List of {@link SkillServletVerifier}.
*/
private transient final List verifiers;
/**
* Instance of type {@link Serializer} initialized with reference to {@link JacksonSerializer}.
*/
private transient final Serializer serializer = new JacksonSerializer();
/**
* Constructor to build an instance of SkillServlet.
* @param skill an Alexa skill instance.
*/
public SkillServlet(final Skill skill) {
List defaultVerifiers = new ArrayList<>();
if (!ServletUtils.isRequestSignatureCheckSystemPropertyDisabled()) {
defaultVerifiers.add(new SkillRequestSignatureVerifier());
}
Long timestampToleranceProperty = ServletUtils.getTimeStampToleranceSystemProperty();
defaultVerifiers.add(new SkillRequestTimestampVerifier(timestampToleranceProperty != null
? timestampToleranceProperty : DEFAULT_TOLERANCE_MILLIS));
this.skill = skill;
this.verifiers = defaultVerifiers;
}
/**
* Constructor to build an instance of SkillServlet.
* @param skill instance of {@link Skill}.
* @param verifiers list of {@link SkillServletVerifier}.
*/
SkillServlet(final Skill skill, final List verifiers) {
this.skill = skill;
this.verifiers = verifiers;
}
/**
* Handles a POST request. Based on the request parameters, invokes the right method on the
* {@code Skill}.
*
* @param request the object that contains the request the client has made of the servlet
* @param response object that contains the response the servlet sends to the client
* @throws IOException if an input or output error is detected when the servlet handles the request
*/
@Override
protected void doPost(final HttpServletRequest request, final HttpServletResponse response)
throws IOException {
byte[] serializedRequestEnvelope = IOUtils.toByteArray(request.getInputStream());
try {
final RequestEnvelope deserializedRequestEnvelope = serializer.deserialize(IOUtils.toString(
serializedRequestEnvelope, ServletConstants.CHARACTER_ENCODING), RequestEnvelope.class);
final AlexaHttpRequest alexaHttpRequest = new ServletRequest(request, serializedRequestEnvelope, deserializedRequestEnvelope);
// Verify the authenticity of the request by executing configured verifiers.
for (SkillServletVerifier verifier : verifiers) {
verifier.verify(alexaHttpRequest);
}
try (ByteArrayOutputStream skillResponse = new ByteArrayOutputStream()) {
handleRequest(new ByteArrayInputStream(serializedRequestEnvelope), skillResponse);
// Generate JSON and send back the response
response.setContentType("application/json");
response.setStatus(HttpServletResponse.SC_OK);
byte[] serializedResponse = skillResponse.toByteArray();
if (skillResponse != null) {
try (OutputStream out = response.getOutputStream()) {
response.setContentLength(serializedResponse.length);
out.write(serializedResponse);
}
}
}
} catch (SecurityException ex) {
int statusCode = HttpServletResponse.SC_BAD_REQUEST;
LOGGER.error("Incoming request failed verification {}", statusCode, ex);
response.sendError(statusCode, ex.getMessage());
} catch (AskSdkException ex) {
int statusCode = HttpServletResponse.SC_INTERNAL_SERVER_ERROR;
LOGGER.error("Exception occurred in doPost, returning status code {}", statusCode, ex);
response.sendError(statusCode, ex.getMessage());
}
}
/**
* This method is the entry point when executing your servlet. The configured
* {@code SkillServlet} determines the type of request and passes the request to
* the configured {@code Skill}.
*
* @param input - input stream of the request.
* @param output - output stream of the response.
* @throws IOException if an input or output error is detected when the servlet handles the request
*/
public final void handleRequest(final InputStream input, final OutputStream output) throws IOException {
byte[] inputBytes = IOUtils.toByteArray(input);
final BaseSkillRequest skillRequest = new BaseSkillRequest(inputBytes);
SkillResponse> skillResponse = skill.execute(skillRequest);
if (skillResponse != null) {
if (skillResponse.isPresent()) {
skillResponse.writeTo(output);
}
} else {
String message = "Unable to find a suitable request handler";
LOGGER.error(message);
throw new AskSdkException(message);
}
}
/**
* Sets a {@code Proxy} object that this servlet may use if Request Signature Verification is enabled.
*
* @param proxy the {@code Proxy} to associate with this servlet.
*/
public void setProxy(final Proxy proxy) {
if (verifiers.removeIf(verifier -> verifier instanceof SkillRequestSignatureVerifier)) {
verifiers.add(new SkillRequestSignatureVerifier(proxy));
}
}
/**
* Method throws an {@link NotSerializableException} if the servlet is not serializable.
* @param in instance of {@link ObjectInputStream}.
* @throws IOException I/O exception.
* @throws ClassNotFoundException cannot a class through its fully-qualified name and can not find its definition on the classpath.
*/
private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException {
throw new NotSerializableException("Skill servlet is not serializable");
}
/**
* Method throws an {@link NotSerializableException} if the servlet is not serializable.
* @param out instance of {@link ObjectOutputStream}.
* @throws IOException I/O exception.
*/
private void writeObject(final ObjectOutputStream out) throws IOException {
throw new NotSerializableException("Skill servlet is not serializable");
}
}