This project aims to solve following problems for AWS API Users:
- Minimize jar-hell you might have when adding aws-java-sdk library (especially useful if you already have any Google API libraries added to your project)
- Unified interface for request/response paging
- Exception categorization
The project consists of two parts:
- Cloud Endpoints based API, that designed to run on AppEngine Flexible Java but can be deployed to any jetty or tomcat.
- Client library generated by Google APIs Client Generator
import com.cloudaware.cloudmine.amazon.ec2.Ec2;
import com.cloudaware.cloudmine.amazon.ec2.model.InstanceAttribute;
import com.cloudaware.cloudmine.amazon.ec2.model.InstanceAttributeResponse;
import com.cloudaware.cloudmine.amazon.ec2.model.Volume;
import com.cloudaware.cloudmine.amazon.ec2.model.VolumesResponse;
import com.google.api.client.googleapis.javanet.GoogleNetHttpTransport;
import com.google.api.client.googleapis.services.json.AbstractGoogleJsonClientRequest;
import com.google.api.client.http.HttpRequest;
import com.google.api.client.http.HttpRequestInitializer;
import com.google.api.client.json.GenericJson;
import com.google.api.client.json.jackson2.JacksonFactory;
import com.google.common.collect.Lists;
import java.io.IOException;
import java.security.GeneralSecurityException;
import java.util.List;
public final class Example {
public static final String ROOT_URL = "https://cloudmine-amazon-dot-yourproject.appspot.com/_ah/api";
public static final String API_KEY = "YOUR_API_KEY";
private final Ec2 client;
public Example() throws GeneralSecurityException, IOException {
this.client = new Ec2.Builder(
GoogleNetHttpTransport.newTrustedTransport(),
JacksonFactory.getDefaultInstance(),
new RequestInitializer()
)
.setApplicationName("Cloudaware")
.setRootUrl(ROOT_URL)
.build();
}
/**
* Use this method if you know that there is not paging for this API call
*/
public static <T extends AbstractGoogleJsonClientRequest<T2>, T2 extends GenericJson> T2 unwrapSingle(final T call) throws IOException {
call.set("prettyPrint", false);
call.set("key", API_KEY);
final T2 response = call.execute();
rethrowException(response);
return response;
}
/**
* Use this method if you know that there paging for this API call or you not sure
*/
public static <T extends AbstractGoogleJsonClientRequest<T2>, T2 extends GenericJson> List<T2> unwrapList(final T call) throws IOException {
call.set("prettyPrint", false);
call.set("key", API_KEY);
final List<T2> out = Lists.newArrayList();
T2 response = call.execute();
rethrowException(response);
out.add(response);
while (response.get("nextPage") != null) {
response = call.set("page", response.get("nextPage")).execute();
rethrowException(response);
out.add(response);
}
return out;
}
/**
* This method will re-throw exception caught by cloudmine-amazon and then serialized into "exception" property of 200 response
*/
private static <T extends AbstractGoogleJsonClientRequest<T2>, T2 extends GenericJson> void rethrowException(final T2 response) {
final Object exceptionObject = response.get("exception");
if (exceptionObject != null) {
if (exceptionObject instanceof GenericJson) {
final GenericJson exception = (GenericJson) exceptionObject;
final Object category = exception.get("category");
final Object className = exception.get("className");
final Object message = exception.get("message");
final Object requestId = exception.get("requestId");
final Object errorCode = exception.get("errorCode");
final Object errorType = exception.get("errorType");
final Object errorMessage = exception.get("errorMessage");
final Object statusCode = exception.get("statusCode");
final Object serviceName = exception.get("serviceName");
throw new AmazonException(
String.valueOf(category),
String.valueOf(className),
String.valueOf(message),
String.valueOf(requestId),
String.valueOf(errorCode),
String.valueOf(errorType),
String.valueOf(errorMessage),
statusCode == null ? null : Integer.valueOf(String.valueOf(statusCode)),
String.valueOf(serviceName)
);
} else {
throw new RuntimeException("response.exception is not GenericJson: " + exceptionObject);
}
}
}
/**
* Use this method to serialize basic AWS credentials to cloudmine-amazon format
*/
public static String basicCredentials(final String accessKey, final String secretKey) {
return accessKey + "::" + secretKey;
}
/**
* Use this method to serialize session credentials to cloudmine-amazon format
*/
public static String sessionCredentials(final String accessKey, final String secretKey, final String sessionToken) {
return accessKey + "::" + secretKey + "::" + sessionToken;
}
/**
* Example 1: list of all EBS Volumes in a specific region
*/
public List<Volume> listVolumes(final String credentials, final String region) throws IOException {
final List<VolumesResponse> response = unwrapList(client.volumes().list(
credentials,
region
));
final List<Volume> out = Lists.newArrayList();
for (final VolumesResponse r : response) {
if (r != null && r.getVolumes() != null) {
out.addAll(r.getVolumes());
}
}
return out;
}
/**
* Example 2: Instance attribute
*/
public InstanceAttribute getInstanceAttribute(final String credentials, final String region, final String instanceId, final String attributeName) throws IOException {
final InstanceAttributeResponse response = unwrapSingle(client.instances().attributes().get(
credentials,
region,
instanceId,
attributeName
));
if (response != null) {
return response.getInstanceAttribute();
}
return null;
}
/**
* Request Initializer to set greater timeout then default
*/
public static class RequestInitializer implements HttpRequestInitializer {
private static final int READ_TIMEOUT = 120000; //milliseconds
@Override
public void initialize(final HttpRequest request) throws IOException {
request.setReadTimeout(READ_TIMEOUT);
}
}
/**
* Same POJO used by cloudmine-amazon to serialize exceptions from AWS
*/
public static final class AmazonException extends RuntimeException {
private final String category;
private final String className;
private final String requestId;
private final String errorCode;
private final String errorType;
private final String errorMessage;
private final Integer statusCode;
private final String serviceName;
public AmazonException(
final String category,
final String className,
final String message,
final String requestId,
final String errorCode,
final String errorType,
final String errorMessage,
final Integer statusCode,
final String serviceName
) {
super(message);
this.category = category;
this.className = className;
this.requestId = requestId;
this.errorCode = errorCode;
this.errorType = errorType;
this.errorMessage = errorMessage;
this.statusCode = statusCode;
this.serviceName = serviceName;
}
public String getCategory() {
return category;
}
public String getClassName() {
return className;
}
public String getRequestId() {
return requestId;
}
public String getErrorCode() {
return errorCode;
}
public String getErrorType() {
return errorType;
}
public String getErrorMessage() {
return errorMessage;
}
public Integer getStatusCode() {
return statusCode;
}
public String getServiceName() {
return serviceName;
}
}
}
mvn clean package -Dappengine.app.id=yourproject -Dendpoints.service.version=2017-04-28r0
Optionally you can override API name completely with -Dendpoints.service.name=yourapi.endpoints.yourproject.cloud.goog
To start dev server:
mvn clean package jetty:run-exploded -Dappengine.app.id=yourproject -Dendpoints.service.version=2017-04-28r0
- Build project - you will create/update
cloudmine-amazon.json
in root folder gcloud service-management deploy cloudmine-amazon.json
- Receive new API Version and update property
endpoints.service.version
mvn clean package appengine:update -Dappengine.app.id=yourproject -Dendpoints.service.version=2017-05-02r0
(2017-05-02r0
- is your new version)
Client library built using apis-client-generator
Each successful build will leave following files and directories:
cloudmine-amazon.json
- OpenAPI (Swagger) definition for Cloud Endpoints service managementdisovery/*.discovery
- discovery documents for API Explorer and library generation
To build client library:
- Build project
- Execute
client/generate.sh && cd target/client && mvn versions:set -DnewVersion=2017-04-28r0
mvn package
ormvn install
- 1.0.77 - IAM, User Tags and Role Tags added
- 1.0.76 - x-google-allow for ESP
- 1.0.75 - CodeDeploy deploymentTarget endpoints added
- 1.0.74 - includeShadowTrails flag added to CloudTrail trails.list
- 1.0.73 - AppMesh added
- 1.0.72 - Cost And Usage Report list added
- 1.0.71 - IAM, MFA Devises list added
- 1.0.70 - Glacier VaultAccessPolicy added
- 1.0.69 - S3 lacking methods added
- 1.0.68 - IAM, added account aliases List
- 1.0.67 - CloudHSM added
- 1.0.66 - EC2 Spot added, NAT Gateways page fix
- 1.0.65 - Kinesis ignore invalid dates
- 1.0.64 - Change google-api-client version
- 1.0.63 - CloudWatchEvents added
- 1.0.62 - AppStream 2.0 added
- 1.0.61 - fix names
- 1.0.60 - EC2 AutoScaling actualized
- 1.0.59 - SES actualized
- 1.0.58 - Application AutoScaling added, AWS Auto Scaling Plans added
- 1.0.57 - DynamoDB DescribeContinuousBackups method added
- 1.0.56 - CloudFront actualized
- 1.0.55 - CloudFormation actualized
- 1.0.54 - EMR Security Configuration Added
- 1.0.53 - Elasticsearch actualized
- 1.0.52 - CloudTrail actualized
- 1.0.51 - IAM GenerateCredentialReport action added
- 1.0.50 - Beanstalk increase page sizes
- 1.0.49 - DynamoDB actualized
- 1.0.48 - Beanstalk actualized
- 1.0.47 - Amazon MQ added
- 1.0.46 - AWS Certificate Manager added
- 1.0.45 - Workspaces actualized
- 1.0.44 - S3 requests added
- 1.0.43 - IAM SSH Public Keys filter added
- 1.0.42 - IAM updated
- 1.0.41 - RDS actualized
- 1.0.40 - Route53 actualized
- 1.0.39 - Redshift actualized
- 1.0.38 - VPC actualized
- 1.0.37 - Cost Explorer added
- 1.0.36 - S3 Server Side Encryption added
- 1.0.35 - GuardDuty added
- 1.0.34 - instanceState filter fro EMR ListInstances
- 1.0.33 - OpsWorks added
- 1.0.32 - SQS and CloudWatch Logs tags
- 1.0.31 - Tag modification refactoring
- 1.0.30 - Batch added
- 1.0.29 - CloudWatch Logs added
- 1.0.28 - API Gateway added
- 1.0.27 - Glue added
- 1.0.26 - StepFunctions added
- 1.0.25 - Athena added
- 1.0.24 - S3 Bucket ACLs added
- 1.0.23 - Lambda tags added
- 1.0.22 - ECR BatchGetImages (repositories.images.list) added
- 1.0.21 - ECR added
- 1.0.20 - CodePipeline added
- 1.0.19 - IoT added, DynamoDB updated
- 1.0.18 - DataPipeline and ElasticTranscoder added
- 1.0.17 - CodeDeploy added
- 1.0.16 - WAF & Shield added
- 1.0.15 - Kinesis added
- 1.0.14 - Service Catalog added
- 1.0.13 - AWS Config added
- 1.0.12 - EMR Instance Fleets added
- 1.0.11 - AutoScaling policies added
- 1.0.10 - CodeCommit added
- 1.0.9 - ELBv2 added
- 1.0.8 - CodeBuild added
- 1.0.7 - CodeStar added
- 1.0.6 - Elasticsearch Service added, EC2 Customer Gateways and NAT Gateways added
- 1.0.5 - Elastic File System added
- 1.0.4 - EC2 Systems Manager added
- 1.0.3 - Exceptions now return API action
- 1.0.2 - Added Redshift Parameter Groups and Security Groups
- 1.0.1 - Added
amazon.sts.callerIdentity.get
- 1.0.0 - Initial release