Compare commits
75 Commits
itsLucario
...
devtools
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7bf0c76554 | ||
|
|
bf9ea8e515 | ||
|
|
39260d1b60 | ||
|
|
7416ec273a | ||
|
|
a38f213cd6 | ||
|
|
895b51f518 | ||
|
|
47f93fa65b | ||
|
|
3b2bdca1f5 | ||
|
|
bbb40924d9 | ||
|
|
ac7df576e3 | ||
|
|
cceeb999cd | ||
|
|
ae3e8c6884 | ||
|
|
409261f790 | ||
|
|
1fe6e5102e | ||
|
|
dbc126e49e | ||
|
|
22193d8c28 | ||
|
|
0e69088cb0 | ||
|
|
8ea03ad675 | ||
|
|
e9f891c9ea | ||
|
|
d6122da3a1 | ||
|
|
b10a25e9fb | ||
|
|
49a8cf0df8 | ||
|
|
c657c7cbd2 | ||
|
|
998bf5f27f | ||
|
|
44ffd6541d | ||
|
|
cee21dc541 | ||
|
|
1d30ae6575 | ||
|
|
532c311ee1 | ||
|
|
414f7fe963 | ||
|
|
228767ec9b | ||
|
|
aa2dfd8f7b | ||
|
|
761ee1ae56 | ||
|
|
b5b7477277 | ||
|
|
b254e97227 | ||
|
|
e754075e7a | ||
|
|
a8414fd370 | ||
|
|
cfcae5e2d6 | ||
|
|
2145f1e823 | ||
|
|
b6606d3173 | ||
|
|
f09271ae2c | ||
|
|
dd264fe6f7 | ||
|
|
62933c822d | ||
|
|
c4575df3a5 | ||
|
|
a36255a670 | ||
|
|
c35f171d32 | ||
|
|
44fe1a4945 | ||
|
|
68c676fc7f | ||
|
|
a6dca60d73 | ||
|
|
635e9d8886 | ||
|
|
1fa58e9379 | ||
|
|
3d7839ca99 | ||
|
|
72411d1c38 | ||
|
|
9b38589de1 | ||
|
|
0d9b022cb9 | ||
|
|
70857b2f22 | ||
|
|
f2dd80ff03 | ||
|
|
7166218d64 | ||
|
|
b61db5648d | ||
|
|
165a2a9ae5 | ||
|
|
f5357af028 | ||
|
|
eefdfd2655 | ||
|
|
bae565cc37 | ||
|
|
e1c0e0eeb7 | ||
|
|
1ee2379ce9 | ||
|
|
ecda1078fc | ||
|
|
f5aae13fed | ||
|
|
0fd1480028 | ||
|
|
e4e5c001ba | ||
|
|
8fb0da2c72 | ||
|
|
9cfd8ed344 | ||
|
|
2b1d8e4682 | ||
|
|
94ea44f5fe | ||
|
|
3c12cfb17b | ||
|
|
4a98c430d4 | ||
|
|
a2692da265 |
36
.github/workflows/ci.yml
vendored
36
.github/workflows/ci.yml
vendored
@@ -1,16 +1,46 @@
|
||||
name: CI
|
||||
|
||||
on: [push]
|
||||
on: [push, pull_request]
|
||||
|
||||
jobs:
|
||||
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
strategy:
|
||||
matrix:
|
||||
# The MODULE environment variable is evaluated in build-all.sh to run a subset
|
||||
# of the builds. This way, multiple modules can be built in parallel.
|
||||
module: [ "module1", "module2", "module3", "module4", "module5" ]
|
||||
|
||||
steps:
|
||||
|
||||
- name: "Checkout sources"
|
||||
uses: actions/checkout@v1
|
||||
|
||||
- name: "Setup Java"
|
||||
uses: actions/setup-java@v1
|
||||
with:
|
||||
java-version: 13
|
||||
- name: "Build all modules"
|
||||
run: chmod 755 build-all.sh && ./build-all.sh
|
||||
|
||||
- name: "Build module ${{ matrix.module }}"
|
||||
env:
|
||||
MODULE: ${{ matrix.module }}
|
||||
# We don't actually need AWS credentials in the tests, but LocalStack
|
||||
# complains if they're not there, so we add dummies to the environment.
|
||||
AWS_ACCESS_KEY_ID: dummy
|
||||
AWS_SECRET_ACCESS_KEY: dummy
|
||||
AWS_REGION: us-east-1
|
||||
run: |
|
||||
chmod 755 build-all.sh && ./build-all.sh $MODULE
|
||||
|
||||
- name: "Zip build reports"
|
||||
if: failure()
|
||||
run: zip -r reports.zip **/**/build/reports
|
||||
|
||||
- uses: actions/upload-artifact@v1
|
||||
name: "Upload build reports"
|
||||
if: failure()
|
||||
with:
|
||||
name: reports
|
||||
path: reports.zip
|
||||
|
||||
15
README.md
15
README.md
@@ -5,17 +5,4 @@
|
||||
This repo contains example projects which show how to use different (not only) Java technologies.
|
||||
The examples are usually accompanied by a blog post on [https://reflectoring.io](https://reflectoring.io).
|
||||
|
||||
See the READMEs in each subdirectory of this repo for more information on each module.
|
||||
|
||||
## Java Modules
|
||||
All Java modules require **Java 11** to compile and run.
|
||||
|
||||
### Building with Gradle
|
||||
|
||||
Each module should be an independent build and can be built by calling `./gradlew clean build` in the module directory.
|
||||
|
||||
All modules are listed in [build-all.sh](build-all.sh) to run in the CI pipeline.
|
||||
|
||||
### Non-Java Modules
|
||||
|
||||
Some folders contain non-Java projects. For those, refer to the README within the module folder.
|
||||
See the READMEs in each subdirectory of this repo for more information on each module.
|
||||
BIN
aws/.DS_Store
vendored
Normal file
BIN
aws/.DS_Store
vendored
Normal file
Binary file not shown.
14
aws/aws-hello-world/README.md
Normal file
14
aws/aws-hello-world/README.md
Normal file
@@ -0,0 +1,14 @@
|
||||
# AWS Hello World
|
||||
|
||||
A simple Spring Boot application you can use to test deployments to AWS (or any other cloud provider, for that matter).
|
||||
|
||||
This application is also available as a Docker image on Docker Hub: [https://hub.docker.com/r/reflectoring/aws-hello-world](https://hub.docker.com/r/reflectoring/aws-hello-world).
|
||||
|
||||
## Blog posts
|
||||
|
||||
Blog posts about this topic:
|
||||
|
||||
* [The AWS Journey Part 1: Deploying Your First Docker Image](https://reflectoring.io/aws-deploy-docker-image-via-web-console/)
|
||||
* [The AWS Journey Part 2: Deploying a Docker Image with AWS CloudFormation](https://reflectoring.io/aws-cloudformation-deploy-docker-image/)
|
||||
* [The AWS Journey Part 3: Connecting a Spring Boot Application to an RDS Instance with CloudFormation](https://reflectoring.io/aws-cloudformation-rds/)
|
||||
* [The AWS Journey Part 4: Zero-Downtime Deployment with CloudFormation and ECS](https://reflectoring.io/aws-cloudformation-ecs-deployment/)
|
||||
@@ -23,3 +23,18 @@ Use the image instead of your real application to test AWS CloudFormation stacks
|
||||
```
|
||||
5. If the Spring Boot application can connect to the database, it will start up sucessfully and serve a message on the endpoint `/hello`.
|
||||
|
||||
# AWS Hello World
|
||||
|
||||
A simple Spring Boot application you can use to test deployments to AWS (or any other cloud provider, for that matter).
|
||||
|
||||
This application is also available as a Docker image on Docker Hub: [https://hub.docker.com/r/reflectoring/aws-hello-world](https://hub.docker.com/r/reflectoring/aws-hello-world).
|
||||
|
||||
## Blog posts
|
||||
|
||||
Blog posts about this topic:
|
||||
|
||||
* [The AWS Journey Part 1: Deploying Your First Docker Image](https://reflectoring.io/aws-deploy-docker-image-via-web-console/)
|
||||
* [The AWS Journey Part 2: Deploying a Docker Image with AWS CloudFormation](https://reflectoring.io/aws-cloudformation-deploy-docker-image/)
|
||||
* [The AWS Journey Part 3: Connecting a Spring Boot Application to an RDS Instance with CloudFormation](https://reflectoring.io/aws-cloudformation-rds/)
|
||||
* [The AWS Journey Part 4: Zero-Downtime Deployment with CloudFormation and ECS](https://reflectoring.io/aws-cloudformation-ecs-deployment/)
|
||||
|
||||
|
||||
@@ -2,7 +2,12 @@
|
||||
|
||||

|
||||
|
||||
# Companion Blog Post
|
||||
## Blog posts
|
||||
|
||||
[The AWS Journey Part 2: Deploying a Docker image from the Command Line with CloudFormation](https://reflectoring.io/aws-cloudformation-deploy-docker-image/)
|
||||
Blog posts about this topic:
|
||||
|
||||
* [The AWS Journey Part 1: Deploying Your First Docker Image](https://reflectoring.io/aws-deploy-docker-image-via-web-console/)
|
||||
* [The AWS Journey Part 2: Deploying a Docker Image with AWS CloudFormation](https://reflectoring.io/aws-cloudformation-deploy-docker-image/)
|
||||
* [The AWS Journey Part 3: Connecting a Spring Boot Application to an RDS Instance with CloudFormation](https://reflectoring.io/aws-cloudformation-rds/)
|
||||
* [The AWS Journey Part 4: Zero-Downtime Deployment with CloudFormation and ECS](https://reflectoring.io/aws-cloudformation-ecs-deployment/)
|
||||
|
||||
|
||||
@@ -2,7 +2,14 @@
|
||||
|
||||

|
||||
|
||||
# Companion Blog Post
|
||||
## Blog posts
|
||||
|
||||
Blog posts about this topic:
|
||||
|
||||
* [The AWS Journey Part 1: Deploying Your First Docker Image](https://reflectoring.io/aws-deploy-docker-image-via-web-console/)
|
||||
* [The AWS Journey Part 2: Deploying a Docker Image with AWS CloudFormation](https://reflectoring.io/aws-cloudformation-deploy-docker-image/)
|
||||
* [The AWS Journey Part 3: Connecting a Spring Boot Application to an RDS Instance with CloudFormation](https://reflectoring.io/aws-cloudformation-rds/)
|
||||
* [The AWS Journey Part 4: Zero-Downtime Deployment with CloudFormation and ECS](https://reflectoring.io/aws-cloudformation-ecs-deployment/)
|
||||
|
||||
|
||||
[The AWS Journey Part 4: Zero-Downtime Deployment with CloudFormation and ECS](https://reflectoring.io/aws-cloudformation-ecs-deployment/)
|
||||
|
||||
|
||||
BIN
aws/localstack/.DS_Store
vendored
Normal file
BIN
aws/localstack/.DS_Store
vendored
Normal file
Binary file not shown.
117
aws/localstack/.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
117
aws/localstack/.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
@@ -0,0 +1,117 @@
|
||||
/*
|
||||
* Copyright 2007-present the original author or authors.
|
||||
*
|
||||
* 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
|
||||
*
|
||||
* https://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.
|
||||
*/
|
||||
import java.net.*;
|
||||
import java.io.*;
|
||||
import java.nio.channels.*;
|
||||
import java.util.Properties;
|
||||
|
||||
public class MavenWrapperDownloader {
|
||||
|
||||
private static final String WRAPPER_VERSION = "0.5.6";
|
||||
/**
|
||||
* Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided.
|
||||
*/
|
||||
private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/"
|
||||
+ WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar";
|
||||
|
||||
/**
|
||||
* Path to the maven-wrapper.properties file, which might contain a downloadUrl property to
|
||||
* use instead of the default one.
|
||||
*/
|
||||
private static final String MAVEN_WRAPPER_PROPERTIES_PATH =
|
||||
".mvn/wrapper/maven-wrapper.properties";
|
||||
|
||||
/**
|
||||
* Path where the maven-wrapper.jar will be saved to.
|
||||
*/
|
||||
private static final String MAVEN_WRAPPER_JAR_PATH =
|
||||
".mvn/wrapper/maven-wrapper.jar";
|
||||
|
||||
/**
|
||||
* Name of the property which should be used to override the default download url for the wrapper.
|
||||
*/
|
||||
private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl";
|
||||
|
||||
public static void main(String args[]) {
|
||||
System.out.println("- Downloader started");
|
||||
File baseDirectory = new File(args[0]);
|
||||
System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath());
|
||||
|
||||
// If the maven-wrapper.properties exists, read it and check if it contains a custom
|
||||
// wrapperUrl parameter.
|
||||
File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH);
|
||||
String url = DEFAULT_DOWNLOAD_URL;
|
||||
if(mavenWrapperPropertyFile.exists()) {
|
||||
FileInputStream mavenWrapperPropertyFileInputStream = null;
|
||||
try {
|
||||
mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile);
|
||||
Properties mavenWrapperProperties = new Properties();
|
||||
mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream);
|
||||
url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url);
|
||||
} catch (IOException e) {
|
||||
System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'");
|
||||
} finally {
|
||||
try {
|
||||
if(mavenWrapperPropertyFileInputStream != null) {
|
||||
mavenWrapperPropertyFileInputStream.close();
|
||||
}
|
||||
} catch (IOException e) {
|
||||
// Ignore ...
|
||||
}
|
||||
}
|
||||
}
|
||||
System.out.println("- Downloading from: " + url);
|
||||
|
||||
File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH);
|
||||
if(!outputFile.getParentFile().exists()) {
|
||||
if(!outputFile.getParentFile().mkdirs()) {
|
||||
System.out.println(
|
||||
"- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'");
|
||||
}
|
||||
}
|
||||
System.out.println("- Downloading to: " + outputFile.getAbsolutePath());
|
||||
try {
|
||||
downloadFileFromURL(url, outputFile);
|
||||
System.out.println("Done");
|
||||
System.exit(0);
|
||||
} catch (Throwable e) {
|
||||
System.out.println("- Error downloading");
|
||||
e.printStackTrace();
|
||||
System.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
private static void downloadFileFromURL(String urlString, File destination) throws Exception {
|
||||
if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) {
|
||||
String username = System.getenv("MVNW_USERNAME");
|
||||
char[] password = System.getenv("MVNW_PASSWORD").toCharArray();
|
||||
Authenticator.setDefault(new Authenticator() {
|
||||
@Override
|
||||
protected PasswordAuthentication getPasswordAuthentication() {
|
||||
return new PasswordAuthentication(username, password);
|
||||
}
|
||||
});
|
||||
}
|
||||
URL website = new URL(urlString);
|
||||
ReadableByteChannel rbc;
|
||||
rbc = Channels.newChannel(website.openStream());
|
||||
FileOutputStream fos = new FileOutputStream(destination);
|
||||
fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE);
|
||||
fos.close();
|
||||
rbc.close();
|
||||
}
|
||||
|
||||
}
|
||||
BIN
aws/localstack/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
BIN
aws/localstack/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
Binary file not shown.
2
aws/localstack/.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
2
aws/localstack/.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip
|
||||
wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar
|
||||
7
aws/localstack/Dockerfile
Normal file
7
aws/localstack/Dockerfile
Normal file
@@ -0,0 +1,7 @@
|
||||
FROM openjdk:8-jre-alpine
|
||||
MAINTAINER pratikdas@yahoo.com
|
||||
RUN addgroup -S spring && adduser -S spring -G spring
|
||||
USER spring:spring
|
||||
ARG JAR_FILE=target/*.jar
|
||||
COPY ${JAR_FILE} app.jar
|
||||
ENTRYPOINT ["java","-jar","/app.jar"]
|
||||
10
aws/localstack/HELP.md
Normal file
10
aws/localstack/HELP.md
Normal file
@@ -0,0 +1,10 @@
|
||||
# Usage Examples of LocalStack
|
||||
|
||||
|
||||
### Reference Documentation
|
||||
|
||||
Two examples of using LocalStack is provided here. Make sure you have Docker installed and docker engine is started. The examples use Java 14. Make changes to pom.xml if you are using a lower version of Java.
|
||||
|
||||
* JUnit Test classes : JUnit Jupiter tests start LocalStack in a Docker container when the test runs and stops the container when test ends.
|
||||
* Spring Boot application: REST API for creating a customer profile in AWS DynamoDB and store profile picture in S3.
|
||||
|
||||
10
aws/localstack/README.md
Normal file
10
aws/localstack/README.md
Normal file
@@ -0,0 +1,10 @@
|
||||
# Localstack
|
||||
|
||||
Example code to test against AWS services locally using Localstack.
|
||||
|
||||
## Blog posts
|
||||
|
||||
Blog posts about this topic:
|
||||
|
||||
* [Local Development with AWS on LocalStack](https://reflectoring.io/aws-localstack/)
|
||||
|
||||
21
aws/localstack/docker-compose.yml
Normal file
21
aws/localstack/docker-compose.yml
Normal file
@@ -0,0 +1,21 @@
|
||||
version: '2.1'
|
||||
|
||||
services:
|
||||
localstack:
|
||||
container_name: "${LOCALSTACK_DOCKER_NAME-localstack_main}"
|
||||
image: localstack/localstack
|
||||
ports:
|
||||
- "4566-4599:4566-4599"
|
||||
- "${PORT_WEB_UI-8080}:${PORT_WEB_UI-8080}"
|
||||
environment:
|
||||
- SERVICES=s3,dynamodb,cloudformation
|
||||
- DEBUG=${DEBUG- }
|
||||
- DATA_DIR=${DATA_DIR- }
|
||||
- PORT_WEB_UI=${PORT_WEB_UI- }
|
||||
- LAMBDA_EXECUTOR=${LAMBDA_EXECUTOR- }
|
||||
- KINESIS_ERROR_PROBABILITY=${KINESIS_ERROR_PROBABILITY- }
|
||||
- DOCKER_HOST=unix:///var/run/docker.sock
|
||||
- HOST_TMP_FOLDER=${TMPDIR}
|
||||
volumes:
|
||||
- "${TMPDIR:-/tmp/localstack}:/tmp/localstack"
|
||||
- "/var/run/docker.sock:/var/run/docker.sock"
|
||||
310
aws/localstack/mvnw
vendored
Executable file
310
aws/localstack/mvnw
vendored
Executable file
@@ -0,0 +1,310 @@
|
||||
#!/bin/sh
|
||||
# ----------------------------------------------------------------------------
|
||||
# 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
|
||||
#
|
||||
# https://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.
|
||||
# ----------------------------------------------------------------------------
|
||||
|
||||
# ----------------------------------------------------------------------------
|
||||
# Maven Start Up Batch script
|
||||
#
|
||||
# Required ENV vars:
|
||||
# ------------------
|
||||
# JAVA_HOME - location of a JDK home dir
|
||||
#
|
||||
# Optional ENV vars
|
||||
# -----------------
|
||||
# M2_HOME - location of maven2's installed home dir
|
||||
# MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||
# e.g. to debug Maven itself, use
|
||||
# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||
# MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||
# ----------------------------------------------------------------------------
|
||||
|
||||
if [ -z "$MAVEN_SKIP_RC" ] ; then
|
||||
|
||||
if [ -f /etc/mavenrc ] ; then
|
||||
. /etc/mavenrc
|
||||
fi
|
||||
|
||||
if [ -f "$HOME/.mavenrc" ] ; then
|
||||
. "$HOME/.mavenrc"
|
||||
fi
|
||||
|
||||
fi
|
||||
|
||||
# OS specific support. $var _must_ be set to either true or false.
|
||||
cygwin=false;
|
||||
darwin=false;
|
||||
mingw=false
|
||||
case "`uname`" in
|
||||
CYGWIN*) cygwin=true ;;
|
||||
MINGW*) mingw=true;;
|
||||
Darwin*) darwin=true
|
||||
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
|
||||
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
|
||||
if [ -z "$JAVA_HOME" ]; then
|
||||
if [ -x "/usr/libexec/java_home" ]; then
|
||||
export JAVA_HOME="`/usr/libexec/java_home`"
|
||||
else
|
||||
export JAVA_HOME="/Library/Java/Home"
|
||||
fi
|
||||
fi
|
||||
;;
|
||||
esac
|
||||
|
||||
if [ -z "$JAVA_HOME" ] ; then
|
||||
if [ -r /etc/gentoo-release ] ; then
|
||||
JAVA_HOME=`java-config --jre-home`
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ -z "$M2_HOME" ] ; then
|
||||
## resolve links - $0 may be a link to maven's home
|
||||
PRG="$0"
|
||||
|
||||
# need this for relative symlinks
|
||||
while [ -h "$PRG" ] ; do
|
||||
ls=`ls -ld "$PRG"`
|
||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||
if expr "$link" : '/.*' > /dev/null; then
|
||||
PRG="$link"
|
||||
else
|
||||
PRG="`dirname "$PRG"`/$link"
|
||||
fi
|
||||
done
|
||||
|
||||
saveddir=`pwd`
|
||||
|
||||
M2_HOME=`dirname "$PRG"`/..
|
||||
|
||||
# make it fully qualified
|
||||
M2_HOME=`cd "$M2_HOME" && pwd`
|
||||
|
||||
cd "$saveddir"
|
||||
# echo Using m2 at $M2_HOME
|
||||
fi
|
||||
|
||||
# For Cygwin, ensure paths are in UNIX format before anything is touched
|
||||
if $cygwin ; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME=`cygpath --unix "$M2_HOME"`
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
|
||||
[ -n "$CLASSPATH" ] &&
|
||||
CLASSPATH=`cygpath --path --unix "$CLASSPATH"`
|
||||
fi
|
||||
|
||||
# For Mingw, ensure paths are in UNIX format before anything is touched
|
||||
if $mingw ; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME="`(cd "$M2_HOME"; pwd)`"
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`"
|
||||
fi
|
||||
|
||||
if [ -z "$JAVA_HOME" ]; then
|
||||
javaExecutable="`which javac`"
|
||||
if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then
|
||||
# readlink(1) is not available as standard on Solaris 10.
|
||||
readLink=`which readlink`
|
||||
if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then
|
||||
if $darwin ; then
|
||||
javaHome="`dirname \"$javaExecutable\"`"
|
||||
javaExecutable="`cd \"$javaHome\" && pwd -P`/javac"
|
||||
else
|
||||
javaExecutable="`readlink -f \"$javaExecutable\"`"
|
||||
fi
|
||||
javaHome="`dirname \"$javaExecutable\"`"
|
||||
javaHome=`expr "$javaHome" : '\(.*\)/bin'`
|
||||
JAVA_HOME="$javaHome"
|
||||
export JAVA_HOME
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ -z "$JAVACMD" ] ; then
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||
else
|
||||
JAVACMD="$JAVA_HOME/bin/java"
|
||||
fi
|
||||
else
|
||||
JAVACMD="`which java`"
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
echo "Error: JAVA_HOME is not defined correctly." >&2
|
||||
echo " We cannot execute $JAVACMD" >&2
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "$JAVA_HOME" ] ; then
|
||||
echo "Warning: JAVA_HOME environment variable is not set."
|
||||
fi
|
||||
|
||||
CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher
|
||||
|
||||
# traverses directory structure from process work directory to filesystem root
|
||||
# first directory with .mvn subdirectory is considered project base directory
|
||||
find_maven_basedir() {
|
||||
|
||||
if [ -z "$1" ]
|
||||
then
|
||||
echo "Path not specified to find_maven_basedir"
|
||||
return 1
|
||||
fi
|
||||
|
||||
basedir="$1"
|
||||
wdir="$1"
|
||||
while [ "$wdir" != '/' ] ; do
|
||||
if [ -d "$wdir"/.mvn ] ; then
|
||||
basedir=$wdir
|
||||
break
|
||||
fi
|
||||
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
|
||||
if [ -d "${wdir}" ]; then
|
||||
wdir=`cd "$wdir/.."; pwd`
|
||||
fi
|
||||
# end of workaround
|
||||
done
|
||||
echo "${basedir}"
|
||||
}
|
||||
|
||||
# concatenates all lines of a file
|
||||
concat_lines() {
|
||||
if [ -f "$1" ]; then
|
||||
echo "$(tr -s '\n' ' ' < "$1")"
|
||||
fi
|
||||
}
|
||||
|
||||
BASE_DIR=`find_maven_basedir "$(pwd)"`
|
||||
if [ -z "$BASE_DIR" ]; then
|
||||
exit 1;
|
||||
fi
|
||||
|
||||
##########################################################################################
|
||||
# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||
# This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||
##########################################################################################
|
||||
if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found .mvn/wrapper/maven-wrapper.jar"
|
||||
fi
|
||||
else
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
|
||||
fi
|
||||
if [ -n "$MVNW_REPOURL" ]; then
|
||||
jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
else
|
||||
jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
fi
|
||||
while IFS="=" read key value; do
|
||||
case "$key" in (wrapperUrl) jarUrl="$value"; break ;;
|
||||
esac
|
||||
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Downloading from: $jarUrl"
|
||||
fi
|
||||
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
|
||||
if $cygwin; then
|
||||
wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"`
|
||||
fi
|
||||
|
||||
if command -v wget > /dev/null; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found wget ... using wget"
|
||||
fi
|
||||
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||
wget "$jarUrl" -O "$wrapperJarPath"
|
||||
else
|
||||
wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath"
|
||||
fi
|
||||
elif command -v curl > /dev/null; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found curl ... using curl"
|
||||
fi
|
||||
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||
curl -o "$wrapperJarPath" "$jarUrl" -f
|
||||
else
|
||||
curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f
|
||||
fi
|
||||
|
||||
else
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Falling back to using Java to download"
|
||||
fi
|
||||
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
|
||||
# For Cygwin, switch paths to Windows format before running javac
|
||||
if $cygwin; then
|
||||
javaClass=`cygpath --path --windows "$javaClass"`
|
||||
fi
|
||||
if [ -e "$javaClass" ]; then
|
||||
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo " - Compiling MavenWrapperDownloader.java ..."
|
||||
fi
|
||||
# Compiling the Java class
|
||||
("$JAVA_HOME/bin/javac" "$javaClass")
|
||||
fi
|
||||
if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||
# Running the downloader
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo " - Running MavenWrapperDownloader.java ..."
|
||||
fi
|
||||
("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR")
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
##########################################################################################
|
||||
# End of extension
|
||||
##########################################################################################
|
||||
|
||||
export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"}
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo $MAVEN_PROJECTBASEDIR
|
||||
fi
|
||||
MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS"
|
||||
|
||||
# For Cygwin, switch paths to Windows format before running java
|
||||
if $cygwin; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME=`cygpath --path --windows "$M2_HOME"`
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"`
|
||||
[ -n "$CLASSPATH" ] &&
|
||||
CLASSPATH=`cygpath --path --windows "$CLASSPATH"`
|
||||
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
|
||||
MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"`
|
||||
fi
|
||||
|
||||
# Provide a "standardized" way to retrieve the CLI args that will
|
||||
# work with both Windows and non-Windows executions.
|
||||
MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@"
|
||||
export MAVEN_CMD_LINE_ARGS
|
||||
|
||||
WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||
|
||||
exec "$JAVACMD" \
|
||||
$MAVEN_OPTS \
|
||||
-classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \
|
||||
"-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \
|
||||
${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@"
|
||||
182
aws/localstack/mvnw.cmd
vendored
Normal file
182
aws/localstack/mvnw.cmd
vendored
Normal file
@@ -0,0 +1,182 @@
|
||||
@REM ----------------------------------------------------------------------------
|
||||
@REM Licensed to the Apache Software Foundation (ASF) under one
|
||||
@REM or more contributor license agreements. See the NOTICE file
|
||||
@REM distributed with this work for additional information
|
||||
@REM regarding copyright ownership. The ASF licenses this file
|
||||
@REM to you under the Apache License, Version 2.0 (the
|
||||
@REM "License"); you may not use this file except in compliance
|
||||
@REM with the License. You may obtain a copy of the License at
|
||||
@REM
|
||||
@REM https://www.apache.org/licenses/LICENSE-2.0
|
||||
@REM
|
||||
@REM Unless required by applicable law or agreed to in writing,
|
||||
@REM software distributed under the License is distributed on an
|
||||
@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||
@REM KIND, either express or implied. See the License for the
|
||||
@REM specific language governing permissions and limitations
|
||||
@REM under the License.
|
||||
@REM ----------------------------------------------------------------------------
|
||||
|
||||
@REM ----------------------------------------------------------------------------
|
||||
@REM Maven Start Up Batch script
|
||||
@REM
|
||||
@REM Required ENV vars:
|
||||
@REM JAVA_HOME - location of a JDK home dir
|
||||
@REM
|
||||
@REM Optional ENV vars
|
||||
@REM M2_HOME - location of maven2's installed home dir
|
||||
@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands
|
||||
@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending
|
||||
@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||
@REM e.g. to debug Maven itself, use
|
||||
@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||
@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||
@REM ----------------------------------------------------------------------------
|
||||
|
||||
@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on'
|
||||
@echo off
|
||||
@REM set title of command window
|
||||
title %0
|
||||
@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on'
|
||||
@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO%
|
||||
|
||||
@REM set %HOME% to equivalent of $HOME
|
||||
if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%")
|
||||
|
||||
@REM Execute a user defined script before this one
|
||||
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre
|
||||
@REM check for pre script, once with legacy .bat ending and once with .cmd ending
|
||||
if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat"
|
||||
if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd"
|
||||
:skipRcPre
|
||||
|
||||
@setlocal
|
||||
|
||||
set ERROR_CODE=0
|
||||
|
||||
@REM To isolate internal variables from possible post scripts, we use another setlocal
|
||||
@setlocal
|
||||
|
||||
@REM ==== START VALIDATION ====
|
||||
if not "%JAVA_HOME%" == "" goto OkJHome
|
||||
|
||||
echo.
|
||||
echo Error: JAVA_HOME not found in your environment. >&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||
echo location of your Java installation. >&2
|
||||
echo.
|
||||
goto error
|
||||
|
||||
:OkJHome
|
||||
if exist "%JAVA_HOME%\bin\java.exe" goto init
|
||||
|
||||
echo.
|
||||
echo Error: JAVA_HOME is set to an invalid directory. >&2
|
||||
echo JAVA_HOME = "%JAVA_HOME%" >&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||
echo location of your Java installation. >&2
|
||||
echo.
|
||||
goto error
|
||||
|
||||
@REM ==== END VALIDATION ====
|
||||
|
||||
:init
|
||||
|
||||
@REM Find the project base dir, i.e. the directory that contains the folder ".mvn".
|
||||
@REM Fallback to current working directory if not found.
|
||||
|
||||
set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR%
|
||||
IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir
|
||||
|
||||
set EXEC_DIR=%CD%
|
||||
set WDIR=%EXEC_DIR%
|
||||
:findBaseDir
|
||||
IF EXIST "%WDIR%"\.mvn goto baseDirFound
|
||||
cd ..
|
||||
IF "%WDIR%"=="%CD%" goto baseDirNotFound
|
||||
set WDIR=%CD%
|
||||
goto findBaseDir
|
||||
|
||||
:baseDirFound
|
||||
set MAVEN_PROJECTBASEDIR=%WDIR%
|
||||
cd "%EXEC_DIR%"
|
||||
goto endDetectBaseDir
|
||||
|
||||
:baseDirNotFound
|
||||
set MAVEN_PROJECTBASEDIR=%EXEC_DIR%
|
||||
cd "%EXEC_DIR%"
|
||||
|
||||
:endDetectBaseDir
|
||||
|
||||
IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig
|
||||
|
||||
@setlocal EnableExtensions EnableDelayedExpansion
|
||||
for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a
|
||||
@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS%
|
||||
|
||||
:endReadAdditionalConfig
|
||||
|
||||
SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
|
||||
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
|
||||
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||
|
||||
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
|
||||
FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
|
||||
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
|
||||
)
|
||||
|
||||
@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||
@REM This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||
if exist %WRAPPER_JAR% (
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Found %WRAPPER_JAR%
|
||||
)
|
||||
) else (
|
||||
if not "%MVNW_REPOURL%" == "" (
|
||||
SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
)
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Couldn't find %WRAPPER_JAR%, downloading it ...
|
||||
echo Downloading from: %DOWNLOAD_URL%
|
||||
)
|
||||
|
||||
powershell -Command "&{"^
|
||||
"$webclient = new-object System.Net.WebClient;"^
|
||||
"if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^
|
||||
"$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^
|
||||
"}"^
|
||||
"[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^
|
||||
"}"
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Finished downloading %WRAPPER_JAR%
|
||||
)
|
||||
)
|
||||
@REM End of extension
|
||||
|
||||
@REM Provide a "standardized" way to retrieve the CLI args that will
|
||||
@REM work with both Windows and non-Windows executions.
|
||||
set MAVEN_CMD_LINE_ARGS=%*
|
||||
|
||||
%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %*
|
||||
if ERRORLEVEL 1 goto error
|
||||
goto end
|
||||
|
||||
:error
|
||||
set ERROR_CODE=1
|
||||
|
||||
:end
|
||||
@endlocal & set ERROR_CODE=%ERROR_CODE%
|
||||
|
||||
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost
|
||||
@REM check for post script, once with legacy .bat ending and once with .cmd ending
|
||||
if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat"
|
||||
if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd"
|
||||
:skipRcPost
|
||||
|
||||
@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on'
|
||||
if "%MAVEN_BATCH_PAUSE%" == "on" pause
|
||||
|
||||
if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE%
|
||||
|
||||
exit /B %ERROR_CODE%
|
||||
79
aws/localstack/pom.xml
Normal file
79
aws/localstack/pom.xml
Normal file
@@ -0,0 +1,79 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
<parent>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-parent</artifactId>
|
||||
<version>2.3.1.RELEASE</version>
|
||||
<relativePath /> <!-- lookup parent from repository -->
|
||||
</parent>
|
||||
<groupId>io.pratik</groupId>
|
||||
<artifactId>customerregistration</artifactId>
|
||||
<version>1.0</version>
|
||||
<name>customerregistration</name>
|
||||
<description>Spring Boot with Dynamodb and S3 to demonstrate LocalStack</description>
|
||||
|
||||
<properties>
|
||||
<java.version>13</java.version>
|
||||
</properties>
|
||||
<dependencyManagement>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>software.amazon.awssdk</groupId>
|
||||
<artifactId>bom</artifactId>
|
||||
<version>2.9.1</version>
|
||||
<type>pom</type>
|
||||
<scope>import</scope>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
</dependencyManagement>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-web</artifactId>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.projectlombok</groupId>
|
||||
<artifactId>lombok</artifactId>
|
||||
<optional>true</optional>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>software.amazon.awssdk</groupId>
|
||||
<artifactId>dynamodb</artifactId>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>software.amazon.awssdk</groupId>
|
||||
<artifactId>s3</artifactId>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>cloud.localstack</groupId>
|
||||
<artifactId>localstack-utils</artifactId>
|
||||
<version>0.2.1</version>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-test</artifactId>
|
||||
<scope>test</scope>
|
||||
<exclusions>
|
||||
<exclusion>
|
||||
<groupId>org.junit.vintage</groupId>
|
||||
<artifactId>junit-vintage-engine</artifactId>
|
||||
</exclusion>
|
||||
</exclusions>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
|
||||
<build>
|
||||
<plugins>
|
||||
<plugin>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-maven-plugin</artifactId>
|
||||
</plugin>
|
||||
</plugins>
|
||||
</build>
|
||||
|
||||
</project>
|
||||
22
aws/localstack/sample.yaml
Normal file
22
aws/localstack/sample.yaml
Normal file
@@ -0,0 +1,22 @@
|
||||
AWSTemplateFormatVersion: "2010-09-09"
|
||||
Description: A sample template for creating a stack with a bucket and a DynamoDB table.
|
||||
Resources:
|
||||
S3BucketForPoc:
|
||||
Type: AWS::S3::Bucket
|
||||
Properties:
|
||||
BucketName: io.pratik.profileimages
|
||||
DynamoDBTable:
|
||||
Type: AWS::DynamoDB::Table
|
||||
Properties:
|
||||
TableName: entities
|
||||
AttributeDefinitions:
|
||||
-
|
||||
AttributeName: "pk"
|
||||
AttributeType: "S"
|
||||
KeySchema:
|
||||
-
|
||||
AttributeName: "pk"
|
||||
KeyType: "HASH"
|
||||
ProvisionedThroughput:
|
||||
ReadCapacityUnits: 5
|
||||
WriteCapacityUnits: 5
|
||||
@@ -0,0 +1,13 @@
|
||||
package io.reflectoring.customerregistration;
|
||||
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
|
||||
@SpringBootApplication
|
||||
public class CustomerregistrationApplication {
|
||||
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(CustomerregistrationApplication.class, args);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,54 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.controllers;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateResponse;
|
||||
import io.reflectoring.customerregistration.dtos.CustomerDto;
|
||||
import io.reflectoring.customerregistration.services.CustomerService;
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.PathVariable;
|
||||
import org.springframework.web.bind.annotation.PostMapping;
|
||||
import org.springframework.web.bind.annotation.RequestBody;
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.ResponseBody;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@RestController
|
||||
@RequestMapping("/customers")
|
||||
@Slf4j
|
||||
public class CustomerController {
|
||||
|
||||
private CustomerService customerService;
|
||||
|
||||
public CustomerController(CustomerService customerService) {
|
||||
super();
|
||||
this.customerService = customerService;
|
||||
}
|
||||
|
||||
@PostMapping("/")
|
||||
@ResponseBody
|
||||
public CustomerCreateResponse registerCustomer(@RequestBody final CustomerCreateRequest request) {
|
||||
String customerID = customerService.createCustomer(request);
|
||||
return CustomerCreateResponse.builder()
|
||||
.customerID(customerID)
|
||||
.build();
|
||||
|
||||
}
|
||||
|
||||
@GetMapping("/{customerID}")
|
||||
@ResponseBody
|
||||
public CustomerDto getCustomerByID(@PathVariable("customerID") String customerID) {
|
||||
log.info("g=fetching customer with id {}", customerID);
|
||||
return customerService.fetchCustomer(customerID);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,20 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.dtos;
|
||||
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Data
|
||||
public class AddressDto {
|
||||
private String premiseNumber;
|
||||
private String streetName;
|
||||
private String city;
|
||||
private String countryName;
|
||||
private String zip;
|
||||
|
||||
}
|
||||
@@ -0,0 +1,26 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.dtos;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
import lombok.Builder;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Data
|
||||
@Builder
|
||||
public class CustomerCreateRequest {
|
||||
|
||||
private String firstName;
|
||||
private String lastName;
|
||||
private String email;
|
||||
private String phoneNumber;
|
||||
private String gender;
|
||||
private String photo;
|
||||
private List<AddressDto> addresses;
|
||||
}
|
||||
@@ -0,0 +1,19 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.dtos;
|
||||
|
||||
import lombok.Builder;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Data
|
||||
@Builder
|
||||
public class CustomerCreateResponse {
|
||||
|
||||
private String customerID;
|
||||
private String error;
|
||||
}
|
||||
@@ -0,0 +1,28 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.dtos;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
import lombok.Builder;
|
||||
import lombok.Data;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Data
|
||||
@Builder
|
||||
public class CustomerDto {
|
||||
private String customerID;
|
||||
private String userName;
|
||||
private String dateOfBirth;
|
||||
private String email;
|
||||
private String firstName;
|
||||
private String lastName;
|
||||
private String phoneNumber;
|
||||
private String gender;
|
||||
private String photo;
|
||||
private List<AddressDto> addresses;
|
||||
}
|
||||
@@ -0,0 +1,12 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.models;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
public class Address {
|
||||
|
||||
}
|
||||
@@ -0,0 +1,12 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.models;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
public class Credentials {
|
||||
|
||||
}
|
||||
@@ -0,0 +1,35 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.models;
|
||||
|
||||
import java.io.Serializable;
|
||||
import java.util.List;
|
||||
|
||||
import lombok.Builder;
|
||||
import lombok.Data;
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
|
||||
@Data
|
||||
@Builder
|
||||
public class Customer implements Serializable{
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
private String userName;
|
||||
private String dateOfBirth;
|
||||
private String email;
|
||||
private String firstName;
|
||||
private String lastName;
|
||||
private String phoneNumber;
|
||||
private String gender;
|
||||
private String photo;
|
||||
private List<Address> addresses;
|
||||
private Credentials password;
|
||||
|
||||
}
|
||||
@@ -0,0 +1,84 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.repositories;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
import java.nio.file.Paths;
|
||||
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import software.amazon.awssdk.core.sync.RequestBody;
|
||||
import software.amazon.awssdk.core.sync.ResponseTransformer;
|
||||
import software.amazon.awssdk.regions.Region;
|
||||
import software.amazon.awssdk.services.s3.S3Client;
|
||||
import software.amazon.awssdk.services.s3.S3ClientBuilder;
|
||||
import software.amazon.awssdk.services.s3.model.GetObjectRequest;
|
||||
import software.amazon.awssdk.services.s3.model.PutObjectRequest;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Service
|
||||
@Slf4j
|
||||
public class CustomerImageStore {
|
||||
private static final Region region = Region.US_EAST_1;
|
||||
private static final String BUCKET_NAME = "io.pratik.profileimages";
|
||||
|
||||
private final String awsEndpoint;
|
||||
|
||||
public CustomerImageStore(@Value("${aws.local.endpoint:#{null}}") String awsEndpoint) {
|
||||
super();
|
||||
this.awsEndpoint = awsEndpoint;
|
||||
}
|
||||
|
||||
private S3Client getS3Client() {
|
||||
S3Client s3 = null;;
|
||||
try {
|
||||
S3ClientBuilder builder = S3Client.builder();
|
||||
// awsEndpoint is set only in local environments
|
||||
if(awsEndpoint != null) {
|
||||
// override aws endpoint with localstack URL in dev environment
|
||||
builder.endpointOverride(new URI(awsEndpoint));
|
||||
}
|
||||
s3 = builder.region(region).build();
|
||||
}catch(URISyntaxException ex) {
|
||||
log.error("Invalid url {}",awsEndpoint);
|
||||
throw new IllegalStateException("Invalid url "+awsEndpoint,ex);
|
||||
}
|
||||
return s3;
|
||||
}
|
||||
|
||||
/**
|
||||
* Fetch profile image from s3 bucket
|
||||
* @param customerDto
|
||||
* @param key
|
||||
* @return customer ID generated using uuid
|
||||
*/
|
||||
public void fetchProfileImage(final CustomerCreateRequest customerDto, String key) {
|
||||
S3Client s3 = getS3Client();
|
||||
if(s3 != null) {
|
||||
GetObjectRequest getObjectRequest = GetObjectRequest.builder().bucket(BUCKET_NAME).key(key).build();
|
||||
s3.getObject(getObjectRequest);
|
||||
s3.getObject(GetObjectRequest.builder().bucket(BUCKET_NAME).key(key).build(),
|
||||
ResponseTransformer.toFile(Paths.get("image"+key)));
|
||||
}
|
||||
}
|
||||
|
||||
public void saveImage(final CustomerCreateRequest customerDto, final String imageKey) {
|
||||
S3Client s3 = getS3Client();
|
||||
if(s3 != null) {
|
||||
// Put Object
|
||||
PutObjectRequest putObjectRequest = PutObjectRequest.builder().bucket(BUCKET_NAME).key(imageKey).build();
|
||||
RequestBody requestBody = RequestBody.fromString(customerDto.getPhoto());
|
||||
s3.putObject(putObjectRequest, requestBody);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,140 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.repositories;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
import java.util.UUID;
|
||||
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import io.reflectoring.customerregistration.dtos.CustomerDto;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import software.amazon.awssdk.regions.Region;
|
||||
import software.amazon.awssdk.services.dynamodb.DynamoDbClient;
|
||||
import software.amazon.awssdk.services.dynamodb.DynamoDbClientBuilder;
|
||||
import software.amazon.awssdk.services.dynamodb.model.AttributeAction;
|
||||
import software.amazon.awssdk.services.dynamodb.model.AttributeValue;
|
||||
import software.amazon.awssdk.services.dynamodb.model.AttributeValueUpdate;
|
||||
import software.amazon.awssdk.services.dynamodb.model.DynamoDbException;
|
||||
import software.amazon.awssdk.services.dynamodb.model.GetItemRequest;
|
||||
import software.amazon.awssdk.services.dynamodb.model.GetItemResponse;
|
||||
import software.amazon.awssdk.services.dynamodb.model.UpdateItemRequest;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Slf4j
|
||||
@Service
|
||||
public class CustomerProfileStore {
|
||||
private static final String TABLE_NAME = "entities";
|
||||
private static final Region region = Region.US_EAST_1;
|
||||
|
||||
private final String awsEndpoint;
|
||||
|
||||
public CustomerProfileStore(@Value("${aws.local.endpoint:#{null}}") String awsEndpoint) {
|
||||
super();
|
||||
this.awsEndpoint = awsEndpoint;
|
||||
}
|
||||
|
||||
private DynamoDbClient getDdbClient() {
|
||||
DynamoDbClient dynamoDB = null;;
|
||||
try {
|
||||
DynamoDbClientBuilder builder = DynamoDbClient.builder();
|
||||
// awsLocalEndpoint is set only in local environments
|
||||
if(awsEndpoint != null) {
|
||||
// override aws endpoint with localstack URL in dev environment
|
||||
builder.endpointOverride(new URI(awsEndpoint));
|
||||
}
|
||||
dynamoDB = builder.region(region).build();
|
||||
}catch(URISyntaxException ex) {
|
||||
log.error("Invalid url {}",awsEndpoint);
|
||||
throw new IllegalStateException("Invalid url "+awsEndpoint,ex);
|
||||
}
|
||||
return dynamoDB;
|
||||
}
|
||||
|
||||
/**
|
||||
* Store profile data in dynamodb table
|
||||
* @param customerDto
|
||||
* @param key
|
||||
* @return customer ID generated using uuid
|
||||
*/
|
||||
public CustomerDto fetchProfile( String customerID) {
|
||||
|
||||
DynamoDbClient ddb = getDdbClient();
|
||||
|
||||
Map<String, AttributeValue> attributeKey = new HashMap<>();
|
||||
String key = "CUSTOMER:"+customerID;
|
||||
attributeKey.put("pk", AttributeValue.builder().s(key).build());
|
||||
GetItemRequest getItemRequest = GetItemRequest.builder().tableName(TABLE_NAME).key(attributeKey).build();
|
||||
GetItemResponse getItemResponse = ddb.getItem(getItemRequest);
|
||||
Map<String, AttributeValue> responseAttributeMap = getItemResponse.item();
|
||||
return CustomerDto.builder()
|
||||
.customerID(customerID)
|
||||
.firstName(responseAttributeMap.get("fname").s())
|
||||
.lastName(responseAttributeMap.get("lname").s())
|
||||
.email(responseAttributeMap.get("email").s())
|
||||
.phoneNumber(responseAttributeMap.get("phone").s())
|
||||
.build();
|
||||
}
|
||||
|
||||
/**
|
||||
* Store profile data in dynamodb table
|
||||
* @param customerDto
|
||||
* @return customer ID generated using uuid
|
||||
*/
|
||||
public String createProfile(final CustomerCreateRequest customerDto) {
|
||||
DynamoDbClient ddb = getDdbClient();
|
||||
|
||||
HashMap<String,AttributeValue> itemKey = new HashMap<String,AttributeValue>();
|
||||
|
||||
String customerID = UUID.randomUUID().toString();
|
||||
String key = "CUSTOMER:"+customerID;
|
||||
itemKey.put("pk", AttributeValue.builder().s(key).build());
|
||||
|
||||
HashMap<String,AttributeValueUpdate> updatedValues =
|
||||
new HashMap<String,AttributeValueUpdate>();
|
||||
|
||||
// Update the column specified by name with updatedVal
|
||||
updatedValues.put("fname", AttributeValueUpdate.builder()
|
||||
.value(AttributeValue.builder().s(customerDto.getFirstName()).build())
|
||||
.action(AttributeAction.PUT)
|
||||
.build());
|
||||
|
||||
updatedValues.put("lname", AttributeValueUpdate.builder()
|
||||
.value(AttributeValue.builder().s(customerDto.getLastName()).build())
|
||||
.action(AttributeAction.PUT)
|
||||
.build());
|
||||
|
||||
updatedValues.put("phone", AttributeValueUpdate.builder()
|
||||
.value(AttributeValue.builder().s(customerDto.getPhoneNumber()).build())
|
||||
.action(AttributeAction.PUT)
|
||||
.build());
|
||||
|
||||
updatedValues.put("email", AttributeValueUpdate.builder()
|
||||
.value(AttributeValue.builder().s(customerDto.getEmail()).build())
|
||||
.action(AttributeAction.PUT)
|
||||
.build());
|
||||
|
||||
UpdateItemRequest request = UpdateItemRequest.builder()
|
||||
.tableName(TABLE_NAME)
|
||||
.key(itemKey)
|
||||
.attributeUpdates(updatedValues)
|
||||
.build();
|
||||
|
||||
try {
|
||||
ddb.updateItem(request);
|
||||
} catch (DynamoDbException e) {
|
||||
}
|
||||
return customerID;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,47 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration.services;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerDto;
|
||||
import io.reflectoring.customerregistration.repositories.CustomerImageStore;
|
||||
import io.reflectoring.customerregistration.repositories.CustomerProfileStore;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Service
|
||||
@Slf4j
|
||||
public class CustomerService {
|
||||
|
||||
|
||||
private CustomerImageStore customerImageStore;
|
||||
private CustomerProfileStore customerProfileStore;
|
||||
|
||||
|
||||
@Autowired
|
||||
public CustomerService(CustomerImageStore customerImageStore, CustomerProfileStore customerProfileStore) {
|
||||
super();
|
||||
this.customerImageStore = customerImageStore;
|
||||
this.customerProfileStore = customerProfileStore;
|
||||
}
|
||||
|
||||
public CustomerDto fetchCustomer(final String customerID) {
|
||||
return customerProfileStore.fetchProfile(customerID);
|
||||
|
||||
}
|
||||
|
||||
|
||||
public String createCustomer(final CustomerCreateRequest request) {
|
||||
String customerKey = customerProfileStore.createProfile(request);
|
||||
customerImageStore.saveImage(request,customerKey);
|
||||
return customerKey;
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
aws.local.endpoint=http://localhost:4566
|
||||
1
aws/localstack/src/main/resources/application.properties
Normal file
1
aws/localstack/src/main/resources/application.properties
Normal file
@@ -0,0 +1 @@
|
||||
server.port=8085
|
||||
@@ -0,0 +1,101 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
package io.reflectoring.customerregistration;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import io.reflectoring.customerregistration.repositories.CustomerImageStore;
|
||||
import org.junit.jupiter.api.AfterEach;
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.junit.jupiter.api.extension.ExtendWith;
|
||||
|
||||
import cloud.localstack.Localstack;
|
||||
import cloud.localstack.docker.LocalstackDockerExtension;
|
||||
import cloud.localstack.docker.annotation.LocalstackDockerProperties;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.springframework.test.context.ActiveProfiles;
|
||||
import software.amazon.awssdk.regions.Region;
|
||||
import software.amazon.awssdk.services.s3.S3Client;
|
||||
import software.amazon.awssdk.services.s3.model.CreateBucketRequest;
|
||||
import software.amazon.awssdk.services.s3.model.GetObjectRequest;
|
||||
import software.amazon.awssdk.services.s3.model.NoSuchKeyException;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Slf4j
|
||||
@ExtendWith(LocalstackDockerExtension.class)
|
||||
@ActiveProfiles("local")
|
||||
@LocalstackDockerProperties(services = { "s3" })
|
||||
class CustomerImageStoreTest {
|
||||
|
||||
private static final Region region = Region.US_EAST_1;
|
||||
private static final String BUCKET_NAME = "io.pratik.profileimages";
|
||||
|
||||
private CustomerImageStore customerImageStore = null;
|
||||
|
||||
/**
|
||||
* @throws java.lang.Exception
|
||||
*/
|
||||
@BeforeEach
|
||||
void setUp() throws Exception {
|
||||
createBucket();
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws java.lang.Exception
|
||||
*/
|
||||
@AfterEach
|
||||
void tearDown() throws Exception {
|
||||
}
|
||||
|
||||
@Test
|
||||
void testStoreImage() {
|
||||
customerImageStore = new CustomerImageStore(Localstack.INSTANCE.getEndpointS3());
|
||||
String photo = "test image";
|
||||
CustomerCreateRequest customerDto = CustomerCreateRequest.builder().firstName("pratik").photo(photo ).build();
|
||||
String imageKey = customerDto.getFirstName()+System.currentTimeMillis();
|
||||
customerImageStore.saveImage(customerDto, imageKey );
|
||||
assertTrue(keyExistsInBucket(imageKey),"Object created");
|
||||
}
|
||||
|
||||
private void createBucket() {
|
||||
URI localEndpoint = null;
|
||||
try {
|
||||
localEndpoint = new URI(Localstack.INSTANCE.getEndpointS3());
|
||||
S3Client s3 = S3Client.builder().endpointOverride(localEndpoint).region(region).build();
|
||||
CreateBucketRequest createBucketRequest = CreateBucketRequest.builder().bucket(BUCKET_NAME).build();
|
||||
s3.createBucket(createBucketRequest );
|
||||
|
||||
} catch (URISyntaxException e) {
|
||||
// TODO Auto-generated catch block
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
private boolean keyExistsInBucket(final String objectKey ) {
|
||||
URI localEndpoint = null;
|
||||
try {
|
||||
localEndpoint = new URI(Localstack.INSTANCE.getEndpointS3());
|
||||
S3Client s3 = S3Client.builder().endpointOverride(localEndpoint).region(region).build();
|
||||
GetObjectRequest getObjectRequest = GetObjectRequest.builder().bucket(BUCKET_NAME).key(objectKey).build();
|
||||
s3.getObject(getObjectRequest);
|
||||
return true;
|
||||
} catch (URISyntaxException e) {
|
||||
log.error("Invalid url {}",localEndpoint);
|
||||
} catch(NoSuchKeyException ex) {
|
||||
log.error("Key does not exist {}", objectKey);
|
||||
}
|
||||
return false;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,103 @@
|
||||
package io.reflectoring.customerregistration;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
|
||||
import io.reflectoring.customerregistration.dtos.CustomerDto;
|
||||
import io.reflectoring.customerregistration.repositories.CustomerProfileStore;
|
||||
import org.junit.jupiter.api.AfterEach;
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.junit.jupiter.api.extension.ExtendWith;
|
||||
|
||||
|
||||
import cloud.localstack.Localstack;
|
||||
import cloud.localstack.docker.LocalstackDockerExtension;
|
||||
import cloud.localstack.docker.annotation.LocalstackDockerProperties;
|
||||
import io.reflectoring.customerregistration.dtos.CustomerCreateRequest;
|
||||
import lombok.extern.slf4j.Slf4j;
|
||||
import org.springframework.test.context.ActiveProfiles;
|
||||
import software.amazon.awssdk.regions.Region;
|
||||
import software.amazon.awssdk.services.dynamodb.DynamoDbClient;
|
||||
import software.amazon.awssdk.services.dynamodb.model.AttributeDefinition;
|
||||
import software.amazon.awssdk.services.dynamodb.model.CreateTableRequest;
|
||||
import software.amazon.awssdk.services.dynamodb.model.KeySchemaElement;
|
||||
import software.amazon.awssdk.services.dynamodb.model.KeyType;
|
||||
import software.amazon.awssdk.services.dynamodb.model.ProvisionedThroughput;
|
||||
import software.amazon.awssdk.services.dynamodb.model.ScalarAttributeType;
|
||||
|
||||
/**
|
||||
* @author Pratik Das
|
||||
*
|
||||
*/
|
||||
@Slf4j
|
||||
@ExtendWith(LocalstackDockerExtension.class)
|
||||
@ActiveProfiles("local")
|
||||
@LocalstackDockerProperties(services = { "dynamodb"})
|
||||
class CustomerProfileStoreTest {
|
||||
|
||||
private static final Region region = Region.US_EAST_1;
|
||||
private static final String TABLE_NAME = "entities";
|
||||
|
||||
private CustomerProfileStore customerProfileStore = null;
|
||||
|
||||
@BeforeEach
|
||||
void setUp() throws Exception {
|
||||
customerProfileStore = new CustomerProfileStore(Localstack.INSTANCE.getEndpointDynamoDB());
|
||||
createTable();
|
||||
}
|
||||
|
||||
@AfterEach
|
||||
void tearDown() throws Exception {
|
||||
}
|
||||
|
||||
@Test
|
||||
void testCreateCustomerProfile() {
|
||||
CustomerCreateRequest customerDto = CustomerCreateRequest.builder()
|
||||
.firstName("pratik")
|
||||
.lastName("das")
|
||||
.phoneNumber("657576")
|
||||
.email("prat@gmail.co")
|
||||
.gender("M")
|
||||
.build();
|
||||
|
||||
String customerID = customerProfileStore.createProfile(customerDto );
|
||||
log.info("customer ID {}", customerID);
|
||||
assertTrue(customerID != null, "Item created");
|
||||
CustomerDto dto = customerProfileStore.fetchProfile(customerID);
|
||||
assertEquals("pratik", dto.getFirstName(), "first name matched");
|
||||
assertEquals("das", dto.getLastName(), "last name matched");
|
||||
}
|
||||
|
||||
private void createTable() {
|
||||
try {
|
||||
DynamoDbClient ddbClient = DynamoDbClient.builder().endpointOverride(new URI(Localstack.INSTANCE.getEndpointDynamoDB())).region(region).build();
|
||||
|
||||
String key = "pk";
|
||||
CreateTableRequest createTableRequest = CreateTableRequest.builder()
|
||||
.attributeDefinitions(AttributeDefinition.builder()
|
||||
.attributeName(key )
|
||||
.attributeType(ScalarAttributeType.S)
|
||||
.build())
|
||||
.keySchema(KeySchemaElement.builder()
|
||||
.attributeName(key)
|
||||
.keyType(KeyType.HASH)
|
||||
.build())
|
||||
.provisionedThroughput(ProvisionedThroughput.builder()
|
||||
.readCapacityUnits(10l)
|
||||
.writeCapacityUnits(10l)
|
||||
.build())
|
||||
.tableName(TABLE_NAME)
|
||||
.build();
|
||||
|
||||
ddbClient.createTable(createTableRequest);
|
||||
} catch (URISyntaxException e) {
|
||||
// TODO Auto-generated catch block
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
152
build-all.sh
152
build-all.sh
@@ -1,5 +1,6 @@
|
||||
#!/bin/bash
|
||||
|
||||
MODULE=$1
|
||||
MAIN_DIR=$PWD
|
||||
|
||||
build_gradle_module() {
|
||||
@@ -81,57 +82,102 @@ build_maven_module() {
|
||||
}
|
||||
}
|
||||
|
||||
build_maven_module "spring-boot/spring-boot-kafka"
|
||||
build_maven_module "resilience4j/retry"
|
||||
build_maven_module "solid/lsp"
|
||||
run_gradle_task "spring-boot/thymeleaf-vue" "npmInstall"
|
||||
build_gradle_module "spring-boot/thymeleaf-vue"
|
||||
build_gradle_module "spring-boot/spring-boot-springdoc"
|
||||
build_maven_module "spring-boot/dependency-injection"
|
||||
build_maven_module "spring-boot/spring-boot-openapi"
|
||||
build_maven_module "spring-boot/data-migration/liquibase"
|
||||
build_gradle_module "spring-boot/boundaries"
|
||||
build_gradle_module "spring-boot/argumentresolver"
|
||||
build_gradle_module "spring-data/spring-data-jdbc-converter"
|
||||
build_gradle_module "solid"
|
||||
build_gradle_module "spring-boot/data-migration/flyway"
|
||||
build_gradle_module "reactive"
|
||||
build_gradle_module "junit/assumptions"
|
||||
build_gradle_module "logging"
|
||||
build_gradle_module "pact/pact-feign-consumer"
|
||||
# currently disabled since the consumer build won't run
|
||||
# build_gradle_module "pact/pact-message-consumer"
|
||||
# build_gradle_module "pact/pact-message-producer"
|
||||
build_gradle_module "pact/pact-spring-provider"
|
||||
build_gradle_module "patterns"
|
||||
build_gradle_module "spring-boot/conditionals"
|
||||
build_gradle_module "spring-boot/configuration"
|
||||
build_gradle_module "spring-boot/mocking"
|
||||
build_gradle_module "spring-boot/modular"
|
||||
build_gradle_module "spring-boot/paging"
|
||||
build_gradle_module "spring-boot/rabbitmq-event-brokering"
|
||||
build_gradle_module "spring-boot/spring-boot-logging"
|
||||
build_gradle_module "spring-boot/spring-boot-testing"
|
||||
build_gradle_module "spring-boot/starter"
|
||||
build_gradle_module "spring-boot/startup"
|
||||
build_gradle_module "spring-boot/static"
|
||||
build_gradle_module "spring-boot/validation"
|
||||
build_gradle_module "spring-boot/profiles"
|
||||
build_gradle_module "spring-boot/password-encoding"
|
||||
build_gradle_module "spring-boot/testcontainers"
|
||||
build_gradle_module "spring-boot/hazelcast/hazelcast-embedded-cache"
|
||||
build_gradle_module "spring-boot/hazelcast/hazelcast-client-server"
|
||||
build_gradle_module "spring-boot/cache"
|
||||
build_gradle_module "spring-cloud/feign-with-spring-data-rest"
|
||||
build_gradle_module "spring-cloud/sleuth-downstream-service"
|
||||
build_gradle_module "spring-cloud/sleuth-upstream-service"
|
||||
build_gradle_module "spring-cloud/spring-cloud-contract-consumer"
|
||||
build_gradle_module "spring-cloud/spring-cloud-contract-provider"
|
||||
build_gradle_module "spring-data/spring-data-rest-associations"
|
||||
build_gradle_module "spring-data/spring-data-rest-springfox"
|
||||
build_gradle_module "tools/jacoco"
|
||||
if [[ "$MODULE" == "module5" ]]
|
||||
then
|
||||
# ADD NEW MODULES HERE
|
||||
# (add new modules above the rest so you get quicker feedback if it fails)
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ ALL MODULES SUCCESSFUL"
|
||||
echo "+++"
|
||||
build_gradle_module "spring-boot/devtools-demo"
|
||||
build_gradle_module "spring-boot/cache"
|
||||
build_gradle_module "spring-boot/bean-lifecycle"
|
||||
build_gradle_module "spring-boot/request-response/client"
|
||||
build_gradle_module "spring-boot/request-response/server"
|
||||
build_gradle_module "spring-boot/hazelcast/hazelcast-embedded-cache"
|
||||
build_gradle_module "spring-boot/hazelcast/hazelcast-client-server"
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ MODULE 5 SUCCESSFUL"
|
||||
echo "+++"
|
||||
fi
|
||||
|
||||
|
||||
if [[ "$MODULE" == "module1" ]]
|
||||
then
|
||||
build_maven_module "spring-boot/spring-boot-kafka"
|
||||
build_gradle_module "spring-boot/spring-boot-springdoc"
|
||||
build_maven_module "spring-boot/dependency-injection"
|
||||
build_maven_module "spring-boot/spring-boot-openapi"
|
||||
build_maven_module "spring-boot/data-migration/liquibase"
|
||||
build_gradle_module "spring-boot/boundaries"
|
||||
build_gradle_module "spring-boot/argumentresolver"
|
||||
build_gradle_module "spring-boot/data-migration/flyway"
|
||||
run_gradle_task "spring-boot/thymeleaf-vue" "clean npmInstall build"
|
||||
build_gradle_module "spring-boot/conditionals"
|
||||
build_gradle_module "spring-boot/configuration"
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ MODULE 1 SUCCESSFUL"
|
||||
echo "+++"
|
||||
fi
|
||||
|
||||
if [[ "$MODULE" == "module2" ]]
|
||||
then
|
||||
build_gradle_module "solid/isp"
|
||||
build_maven_module "solid/lsp"
|
||||
build_maven_module "resilience4j/retry"
|
||||
build_maven_module "resilience4j/ratelimiter"
|
||||
build_gradle_module "spring-data/spring-data-jdbc-converter"
|
||||
build_gradle_module "reactive"
|
||||
build_gradle_module "junit/assumptions"
|
||||
build_gradle_module "logging"
|
||||
build_gradle_module "pact/pact-feign-consumer"
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ MODULE 2 SUCCESSFUL"
|
||||
echo "+++"
|
||||
fi
|
||||
|
||||
if [[ "$MODULE" == "module3" ]]
|
||||
then
|
||||
build_maven_module "aws/localstack"
|
||||
build_gradle_module "pact/pact-spring-provider"
|
||||
build_gradle_module "patterns"
|
||||
build_gradle_module "spring-cloud/feign-with-spring-data-rest"
|
||||
build_gradle_module "spring-cloud/sleuth-downstream-service"
|
||||
build_gradle_module "spring-cloud/sleuth-upstream-service"
|
||||
build_gradle_module "spring-cloud/spring-cloud-contract-provider" # has to run before consumer
|
||||
build_gradle_module "spring-cloud/spring-cloud-contract-consumer"
|
||||
build_gradle_module "spring-data/spring-data-rest-associations"
|
||||
build_gradle_module "spring-data/spring-data-rest-springfox"
|
||||
build_gradle_module "tools/jacoco"
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ MODULE 3 SUCCESSFUL"
|
||||
echo "+++"
|
||||
fi
|
||||
|
||||
if [[ "$MODULE" == "module4" ]]
|
||||
then
|
||||
build_gradle_module "spring-boot/mocking"
|
||||
build_gradle_module "spring-boot/modular"
|
||||
build_gradle_module "spring-boot/paging"
|
||||
build_gradle_module "spring-boot/rabbitmq-event-brokering"
|
||||
build_gradle_module "spring-boot/spring-boot-logging"
|
||||
build_gradle_module "spring-boot/spring-boot-testing"
|
||||
build_gradle_module "spring-boot/starter"
|
||||
build_gradle_module "spring-boot/startup"
|
||||
build_gradle_module "spring-boot/static"
|
||||
build_gradle_module "spring-boot/validation"
|
||||
build_gradle_module "spring-boot/profiles"
|
||||
build_gradle_module "spring-boot/password-encoding"
|
||||
build_gradle_module "spring-boot/testcontainers"
|
||||
|
||||
echo ""
|
||||
echo "+++"
|
||||
echo "+++ MODULE 4 SUCCESSFUL"
|
||||
echo "+++"
|
||||
fi
|
||||
@@ -1,3 +1,7 @@
|
||||
# Examples with JUnit 4 and JUnit 5
|
||||
# Examples for assumptions with JUnit 4 and 5
|
||||
|
||||
Have a look at [the code](/src/test/java/com/example/demo/)
|
||||
Have a look at [the code](/src/test/java/com/example/demo/).
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [Assumptions and Conditional Test Execution with JUnit 4 and 5](https://reflectoring.io/conditional-junit4-junit5-tests/)
|
||||
|
||||
0
logging/gradlew
vendored
Normal file → Executable file
0
logging/gradlew
vendored
Normal file → Executable file
0
pact/pact-feign-consumer/gradlew
vendored
Normal file → Executable file
0
pact/pact-feign-consumer/gradlew
vendored
Normal file → Executable file
0
pact/pact-spring-provider/gradlew
vendored
Normal file → Executable file
0
pact/pact-spring-provider/gradlew
vendored
Normal file → Executable file
0
patterns/gradlew
vendored
Normal file → Executable file
0
patterns/gradlew
vendored
Normal file → Executable file
16
pull_request_template.md
Normal file
16
pull_request_template.md
Normal file
@@ -0,0 +1,16 @@
|
||||
Please review the changes you did in this PR and check these points to make sure that the code examples are of high quality:
|
||||
|
||||
- [ ] the source branch for this PR is NOT the master branch of your fork (please create a dedicated branch for the PR)
|
||||
- [ ] the PR doesn't contain code that is irrelevant to the topic of the article
|
||||
- [ ] the PR doesn't contain any other files that are irrelevant to the topic of the article
|
||||
- [ ] the PR contains a README.md file that explains in a sentence or two what the code example is about and links to the article on reflectoring.io (if you added to an existing module, please add the link to the existing README.md file)
|
||||
- [ ] the code is covered with tests where it makes sense
|
||||
- [ ] if you have created a new Maven or Gradle module, please add it to the CI build (search for "# ADD NEW MODULES HERE" in `build-all.sh`)
|
||||
- [ ] used the latest Gradle or Maven version
|
||||
- [ ] used Java 11 or later
|
||||
- [ ] the [GitHub Actions build](https://github.com/thombergs/code-examples/actions?query=workflow%3ACI) is successful for your PR branch
|
||||
|
||||
**Feel free to raise the PR even if the checklist hasn't been worked through, yet**. But please only flag it to be reviewed after going through the checklist above.
|
||||
|
||||
Also feel free to reach out if you have any trouble with any of the steps.
|
||||
|
||||
7
reactive/README.md
Normal file
7
reactive/README.md
Normal file
@@ -0,0 +1,7 @@
|
||||
# Reactive code examples
|
||||
|
||||
Some code examples using reactive code.
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [Reactive Multi-Threading with RxJava - Pitfalls and Solutions](https://reflectoring.io/rxjava-reactive-batch-processing/)
|
||||
117
resilience4j/ratelimiter/.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
117
resilience4j/ratelimiter/.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
@@ -0,0 +1,117 @@
|
||||
/*
|
||||
* Copyright 2007-present the original author or authors.
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
import java.net.*;
|
||||
import java.io.*;
|
||||
import java.nio.channels.*;
|
||||
import java.util.Properties;
|
||||
|
||||
public class MavenWrapperDownloader {
|
||||
|
||||
private static final String WRAPPER_VERSION = "0.5.6";
|
||||
/**
|
||||
* Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided.
|
||||
*/
|
||||
private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/"
|
||||
+ WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar";
|
||||
|
||||
/**
|
||||
* Path to the maven-wrapper.properties file, which might contain a downloadUrl property to
|
||||
* use instead of the default one.
|
||||
*/
|
||||
private static final String MAVEN_WRAPPER_PROPERTIES_PATH =
|
||||
".mvn/wrapper/maven-wrapper.properties";
|
||||
|
||||
/**
|
||||
* Path where the maven-wrapper.jar will be saved to.
|
||||
*/
|
||||
private static final String MAVEN_WRAPPER_JAR_PATH =
|
||||
".mvn/wrapper/maven-wrapper.jar";
|
||||
|
||||
/**
|
||||
* Name of the property which should be used to override the default download url for the wrapper.
|
||||
*/
|
||||
private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl";
|
||||
|
||||
public static void main(String args[]) {
|
||||
System.out.println("- Downloader started");
|
||||
File baseDirectory = new File(args[0]);
|
||||
System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath());
|
||||
|
||||
// If the maven-wrapper.properties exists, read it and check if it contains a custom
|
||||
// wrapperUrl parameter.
|
||||
File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH);
|
||||
String url = DEFAULT_DOWNLOAD_URL;
|
||||
if(mavenWrapperPropertyFile.exists()) {
|
||||
FileInputStream mavenWrapperPropertyFileInputStream = null;
|
||||
try {
|
||||
mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile);
|
||||
Properties mavenWrapperProperties = new Properties();
|
||||
mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream);
|
||||
url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url);
|
||||
} catch (IOException e) {
|
||||
System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'");
|
||||
} finally {
|
||||
try {
|
||||
if(mavenWrapperPropertyFileInputStream != null) {
|
||||
mavenWrapperPropertyFileInputStream.close();
|
||||
}
|
||||
} catch (IOException e) {
|
||||
// Ignore ...
|
||||
}
|
||||
}
|
||||
}
|
||||
System.out.println("- Downloading from: " + url);
|
||||
|
||||
File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH);
|
||||
if(!outputFile.getParentFile().exists()) {
|
||||
if(!outputFile.getParentFile().mkdirs()) {
|
||||
System.out.println(
|
||||
"- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'");
|
||||
}
|
||||
}
|
||||
System.out.println("- Downloading to: " + outputFile.getAbsolutePath());
|
||||
try {
|
||||
downloadFileFromURL(url, outputFile);
|
||||
System.out.println("Done");
|
||||
System.exit(0);
|
||||
} catch (Throwable e) {
|
||||
System.out.println("- Error downloading");
|
||||
e.printStackTrace();
|
||||
System.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
private static void downloadFileFromURL(String urlString, File destination) throws Exception {
|
||||
if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) {
|
||||
String username = System.getenv("MVNW_USERNAME");
|
||||
char[] password = System.getenv("MVNW_PASSWORD").toCharArray();
|
||||
Authenticator.setDefault(new Authenticator() {
|
||||
@Override
|
||||
protected PasswordAuthentication getPasswordAuthentication() {
|
||||
return new PasswordAuthentication(username, password);
|
||||
}
|
||||
});
|
||||
}
|
||||
URL website = new URL(urlString);
|
||||
ReadableByteChannel rbc;
|
||||
rbc = Channels.newChannel(website.openStream());
|
||||
FileOutputStream fos = new FileOutputStream(destination);
|
||||
fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE);
|
||||
fos.close();
|
||||
rbc.close();
|
||||
}
|
||||
|
||||
}
|
||||
BIN
resilience4j/ratelimiter/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
BIN
resilience4j/ratelimiter/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
Binary file not shown.
2
resilience4j/ratelimiter/.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
2
resilience4j/ratelimiter/.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip
|
||||
wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar
|
||||
10
resilience4j/ratelimiter/README.md
Normal file
10
resilience4j/ratelimiter/README.md
Normal file
@@ -0,0 +1,10 @@
|
||||
# Rate limiting with Resilience4J
|
||||
|
||||
Run the Examples program.
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [Implementing Rate Limiting with Resilience4j](https://reflectoring.io/rate-limiting-with-resilience4j)
|
||||
|
||||
|
||||
|
||||
310
resilience4j/ratelimiter/mvnw
vendored
Executable file
310
resilience4j/ratelimiter/mvnw
vendored
Executable file
@@ -0,0 +1,310 @@
|
||||
#!/bin/sh
|
||||
# ----------------------------------------------------------------------------
|
||||
# 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.
|
||||
# ----------------------------------------------------------------------------
|
||||
|
||||
# ----------------------------------------------------------------------------
|
||||
# Maven Start Up Batch script
|
||||
#
|
||||
# Required ENV vars:
|
||||
# ------------------
|
||||
# JAVA_HOME - location of a JDK home dir
|
||||
#
|
||||
# Optional ENV vars
|
||||
# -----------------
|
||||
# M2_HOME - location of maven2's installed home dir
|
||||
# MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||
# e.g. to debug Maven itself, use
|
||||
# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||
# MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||
# ----------------------------------------------------------------------------
|
||||
|
||||
if [ -z "$MAVEN_SKIP_RC" ] ; then
|
||||
|
||||
if [ -f /etc/mavenrc ] ; then
|
||||
. /etc/mavenrc
|
||||
fi
|
||||
|
||||
if [ -f "$HOME/.mavenrc" ] ; then
|
||||
. "$HOME/.mavenrc"
|
||||
fi
|
||||
|
||||
fi
|
||||
|
||||
# OS specific support. $var _must_ be set to either true or false.
|
||||
cygwin=false;
|
||||
darwin=false;
|
||||
mingw=false
|
||||
case "`uname`" in
|
||||
CYGWIN*) cygwin=true ;;
|
||||
MINGW*) mingw=true;;
|
||||
Darwin*) darwin=true
|
||||
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
|
||||
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
|
||||
if [ -z "$JAVA_HOME" ]; then
|
||||
if [ -x "/usr/libexec/java_home" ]; then
|
||||
export JAVA_HOME="`/usr/libexec/java_home`"
|
||||
else
|
||||
export JAVA_HOME="/Library/Java/Home"
|
||||
fi
|
||||
fi
|
||||
;;
|
||||
esac
|
||||
|
||||
if [ -z "$JAVA_HOME" ] ; then
|
||||
if [ -r /etc/gentoo-release ] ; then
|
||||
JAVA_HOME=`java-config --jre-home`
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ -z "$M2_HOME" ] ; then
|
||||
## resolve links - $0 may be a link to maven's home
|
||||
PRG="$0"
|
||||
|
||||
# need this for relative symlinks
|
||||
while [ -h "$PRG" ] ; do
|
||||
ls=`ls -ld "$PRG"`
|
||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||
if expr "$link" : '/.*' > /dev/null; then
|
||||
PRG="$link"
|
||||
else
|
||||
PRG="`dirname "$PRG"`/$link"
|
||||
fi
|
||||
done
|
||||
|
||||
saveddir=`pwd`
|
||||
|
||||
M2_HOME=`dirname "$PRG"`/..
|
||||
|
||||
# make it fully qualified
|
||||
M2_HOME=`cd "$M2_HOME" && pwd`
|
||||
|
||||
cd "$saveddir"
|
||||
# echo Using m2 at $M2_HOME
|
||||
fi
|
||||
|
||||
# For Cygwin, ensure paths are in UNIX format before anything is touched
|
||||
if $cygwin ; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME=`cygpath --unix "$M2_HOME"`
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
|
||||
[ -n "$CLASSPATH" ] &&
|
||||
CLASSPATH=`cygpath --path --unix "$CLASSPATH"`
|
||||
fi
|
||||
|
||||
# For Mingw, ensure paths are in UNIX format before anything is touched
|
||||
if $mingw ; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME="`(cd "$M2_HOME"; pwd)`"
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`"
|
||||
fi
|
||||
|
||||
if [ -z "$JAVA_HOME" ]; then
|
||||
javaExecutable="`which javac`"
|
||||
if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then
|
||||
# readlink(1) is not available as standard on Solaris 10.
|
||||
readLink=`which readlink`
|
||||
if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then
|
||||
if $darwin ; then
|
||||
javaHome="`dirname \"$javaExecutable\"`"
|
||||
javaExecutable="`cd \"$javaHome\" && pwd -P`/javac"
|
||||
else
|
||||
javaExecutable="`readlink -f \"$javaExecutable\"`"
|
||||
fi
|
||||
javaHome="`dirname \"$javaExecutable\"`"
|
||||
javaHome=`expr "$javaHome" : '\(.*\)/bin'`
|
||||
JAVA_HOME="$javaHome"
|
||||
export JAVA_HOME
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ -z "$JAVACMD" ] ; then
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||
else
|
||||
JAVACMD="$JAVA_HOME/bin/java"
|
||||
fi
|
||||
else
|
||||
JAVACMD="`which java`"
|
||||
fi
|
||||
fi
|
||||
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
echo "Error: JAVA_HOME is not defined correctly." >&2
|
||||
echo " We cannot execute $JAVACMD" >&2
|
||||
exit 1
|
||||
fi
|
||||
|
||||
if [ -z "$JAVA_HOME" ] ; then
|
||||
echo "Warning: JAVA_HOME environment variable is not set."
|
||||
fi
|
||||
|
||||
CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher
|
||||
|
||||
# traverses directory structure from process work directory to filesystem root
|
||||
# first directory with .mvn subdirectory is considered project base directory
|
||||
find_maven_basedir() {
|
||||
|
||||
if [ -z "$1" ]
|
||||
then
|
||||
echo "Path not specified to find_maven_basedir"
|
||||
return 1
|
||||
fi
|
||||
|
||||
basedir="$1"
|
||||
wdir="$1"
|
||||
while [ "$wdir" != '/' ] ; do
|
||||
if [ -d "$wdir"/.mvn ] ; then
|
||||
basedir=$wdir
|
||||
break
|
||||
fi
|
||||
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
|
||||
if [ -d "${wdir}" ]; then
|
||||
wdir=`cd "$wdir/.."; pwd`
|
||||
fi
|
||||
# end of workaround
|
||||
done
|
||||
echo "${basedir}"
|
||||
}
|
||||
|
||||
# concatenates all lines of a file
|
||||
concat_lines() {
|
||||
if [ -f "$1" ]; then
|
||||
echo "$(tr -s '\n' ' ' < "$1")"
|
||||
fi
|
||||
}
|
||||
|
||||
BASE_DIR=`find_maven_basedir "$(pwd)"`
|
||||
if [ -z "$BASE_DIR" ]; then
|
||||
exit 1;
|
||||
fi
|
||||
|
||||
##########################################################################################
|
||||
# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||
# This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||
##########################################################################################
|
||||
if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found .mvn/wrapper/maven-wrapper.jar"
|
||||
fi
|
||||
else
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
|
||||
fi
|
||||
if [ -n "$MVNW_REPOURL" ]; then
|
||||
jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
else
|
||||
jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
fi
|
||||
while IFS="=" read key value; do
|
||||
case "$key" in (wrapperUrl) jarUrl="$value"; break ;;
|
||||
esac
|
||||
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Downloading from: $jarUrl"
|
||||
fi
|
||||
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
|
||||
if $cygwin; then
|
||||
wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"`
|
||||
fi
|
||||
|
||||
if command -v wget > /dev/null; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found wget ... using wget"
|
||||
fi
|
||||
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||
wget "$jarUrl" -O "$wrapperJarPath"
|
||||
else
|
||||
wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath"
|
||||
fi
|
||||
elif command -v curl > /dev/null; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Found curl ... using curl"
|
||||
fi
|
||||
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
|
||||
curl -o "$wrapperJarPath" "$jarUrl" -f
|
||||
else
|
||||
curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f
|
||||
fi
|
||||
|
||||
else
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo "Falling back to using Java to download"
|
||||
fi
|
||||
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
|
||||
# For Cygwin, switch paths to Windows format before running javac
|
||||
if $cygwin; then
|
||||
javaClass=`cygpath --path --windows "$javaClass"`
|
||||
fi
|
||||
if [ -e "$javaClass" ]; then
|
||||
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo " - Compiling MavenWrapperDownloader.java ..."
|
||||
fi
|
||||
# Compiling the Java class
|
||||
("$JAVA_HOME/bin/javac" "$javaClass")
|
||||
fi
|
||||
if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
|
||||
# Running the downloader
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo " - Running MavenWrapperDownloader.java ..."
|
||||
fi
|
||||
("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR")
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
##########################################################################################
|
||||
# End of extension
|
||||
##########################################################################################
|
||||
|
||||
export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"}
|
||||
if [ "$MVNW_VERBOSE" = true ]; then
|
||||
echo $MAVEN_PROJECTBASEDIR
|
||||
fi
|
||||
MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS"
|
||||
|
||||
# For Cygwin, switch paths to Windows format before running java
|
||||
if $cygwin; then
|
||||
[ -n "$M2_HOME" ] &&
|
||||
M2_HOME=`cygpath --path --windows "$M2_HOME"`
|
||||
[ -n "$JAVA_HOME" ] &&
|
||||
JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"`
|
||||
[ -n "$CLASSPATH" ] &&
|
||||
CLASSPATH=`cygpath --path --windows "$CLASSPATH"`
|
||||
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
|
||||
MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"`
|
||||
fi
|
||||
|
||||
# Provide a "standardized" way to retrieve the CLI args that will
|
||||
# work with both Windows and non-Windows executions.
|
||||
MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@"
|
||||
export MAVEN_CMD_LINE_ARGS
|
||||
|
||||
WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||
|
||||
exec "$JAVACMD" \
|
||||
$MAVEN_OPTS \
|
||||
-classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \
|
||||
"-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \
|
||||
${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@"
|
||||
182
resilience4j/ratelimiter/mvnw.cmd
vendored
Normal file
182
resilience4j/ratelimiter/mvnw.cmd
vendored
Normal file
@@ -0,0 +1,182 @@
|
||||
@REM ----------------------------------------------------------------------------
|
||||
@REM Licensed to the Apache Software Foundation (ASF) under one
|
||||
@REM or more contributor license agreements. See the NOTICE file
|
||||
@REM distributed with this work for additional information
|
||||
@REM regarding copyright ownership. The ASF licenses this file
|
||||
@REM to you under the Apache License, Version 2.0 (the
|
||||
@REM "License"); you may not use this file except in compliance
|
||||
@REM with the License. You may obtain a copy of the License at
|
||||
@REM
|
||||
@REM http://www.apache.org/licenses/LICENSE-2.0
|
||||
@REM
|
||||
@REM Unless required by applicable law or agreed to in writing,
|
||||
@REM software distributed under the License is distributed on an
|
||||
@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||
@REM KIND, either express or implied. See the License for the
|
||||
@REM specific language governing permissions and limitations
|
||||
@REM under the License.
|
||||
@REM ----------------------------------------------------------------------------
|
||||
|
||||
@REM ----------------------------------------------------------------------------
|
||||
@REM Maven Start Up Batch script
|
||||
@REM
|
||||
@REM Required ENV vars:
|
||||
@REM JAVA_HOME - location of a JDK home dir
|
||||
@REM
|
||||
@REM Optional ENV vars
|
||||
@REM M2_HOME - location of maven2's installed home dir
|
||||
@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands
|
||||
@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending
|
||||
@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven
|
||||
@REM e.g. to debug Maven itself, use
|
||||
@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
|
||||
@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files
|
||||
@REM ----------------------------------------------------------------------------
|
||||
|
||||
@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on'
|
||||
@echo off
|
||||
@REM set title of command window
|
||||
title %0
|
||||
@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on'
|
||||
@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO%
|
||||
|
||||
@REM set %HOME% to equivalent of $HOME
|
||||
if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%")
|
||||
|
||||
@REM Execute a user defined script before this one
|
||||
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre
|
||||
@REM check for pre script, once with legacy .bat ending and once with .cmd ending
|
||||
if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat"
|
||||
if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd"
|
||||
:skipRcPre
|
||||
|
||||
@setlocal
|
||||
|
||||
set ERROR_CODE=0
|
||||
|
||||
@REM To isolate internal variables from possible post scripts, we use another setlocal
|
||||
@setlocal
|
||||
|
||||
@REM ==== START VALIDATION ====
|
||||
if not "%JAVA_HOME%" == "" goto OkJHome
|
||||
|
||||
echo.
|
||||
echo Error: JAVA_HOME not found in your environment. >&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||
echo location of your Java installation. >&2
|
||||
echo.
|
||||
goto error
|
||||
|
||||
:OkJHome
|
||||
if exist "%JAVA_HOME%\bin\java.exe" goto init
|
||||
|
||||
echo.
|
||||
echo Error: JAVA_HOME is set to an invalid directory. >&2
|
||||
echo JAVA_HOME = "%JAVA_HOME%" >&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the >&2
|
||||
echo location of your Java installation. >&2
|
||||
echo.
|
||||
goto error
|
||||
|
||||
@REM ==== END VALIDATION ====
|
||||
|
||||
:init
|
||||
|
||||
@REM Find the project base dir, i.e. the directory that contains the folder ".mvn".
|
||||
@REM Fallback to current working directory if not found.
|
||||
|
||||
set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR%
|
||||
IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir
|
||||
|
||||
set EXEC_DIR=%CD%
|
||||
set WDIR=%EXEC_DIR%
|
||||
:findBaseDir
|
||||
IF EXIST "%WDIR%"\.mvn goto baseDirFound
|
||||
cd ..
|
||||
IF "%WDIR%"=="%CD%" goto baseDirNotFound
|
||||
set WDIR=%CD%
|
||||
goto findBaseDir
|
||||
|
||||
:baseDirFound
|
||||
set MAVEN_PROJECTBASEDIR=%WDIR%
|
||||
cd "%EXEC_DIR%"
|
||||
goto endDetectBaseDir
|
||||
|
||||
:baseDirNotFound
|
||||
set MAVEN_PROJECTBASEDIR=%EXEC_DIR%
|
||||
cd "%EXEC_DIR%"
|
||||
|
||||
:endDetectBaseDir
|
||||
|
||||
IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig
|
||||
|
||||
@setlocal EnableExtensions EnableDelayedExpansion
|
||||
for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a
|
||||
@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS%
|
||||
|
||||
:endReadAdditionalConfig
|
||||
|
||||
SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
|
||||
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
|
||||
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
|
||||
|
||||
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
|
||||
FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
|
||||
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
|
||||
)
|
||||
|
||||
@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
|
||||
@REM This allows using the maven wrapper in projects that prohibit checking in binary data.
|
||||
if exist %WRAPPER_JAR% (
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Found %WRAPPER_JAR%
|
||||
)
|
||||
) else (
|
||||
if not "%MVNW_REPOURL%" == "" (
|
||||
SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
|
||||
)
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Couldn't find %WRAPPER_JAR%, downloading it ...
|
||||
echo Downloading from: %DOWNLOAD_URL%
|
||||
)
|
||||
|
||||
powershell -Command "&{"^
|
||||
"$webclient = new-object System.Net.WebClient;"^
|
||||
"if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^
|
||||
"$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^
|
||||
"}"^
|
||||
"[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^
|
||||
"}"
|
||||
if "%MVNW_VERBOSE%" == "true" (
|
||||
echo Finished downloading %WRAPPER_JAR%
|
||||
)
|
||||
)
|
||||
@REM End of extension
|
||||
|
||||
@REM Provide a "standardized" way to retrieve the CLI args that will
|
||||
@REM work with both Windows and non-Windows executions.
|
||||
set MAVEN_CMD_LINE_ARGS=%*
|
||||
|
||||
%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %*
|
||||
if ERRORLEVEL 1 goto error
|
||||
goto end
|
||||
|
||||
:error
|
||||
set ERROR_CODE=1
|
||||
|
||||
:end
|
||||
@endlocal & set ERROR_CODE=%ERROR_CODE%
|
||||
|
||||
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost
|
||||
@REM check for post script, once with legacy .bat ending and once with .cmd ending
|
||||
if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat"
|
||||
if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd"
|
||||
:skipRcPost
|
||||
|
||||
@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on'
|
||||
if "%MAVEN_BATCH_PAUSE%" == "on" pause
|
||||
|
||||
if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE%
|
||||
|
||||
exit /B %ERROR_CODE%
|
||||
119
resilience4j/ratelimiter/pom.xml
Normal file
119
resilience4j/ratelimiter/pom.xml
Normal file
@@ -0,0 +1,119 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<groupId>io.reflectoring.resilience4j.ratelimit</groupId>
|
||||
<artifactId>ratelimiter</artifactId>
|
||||
<version>1.0-SNAPSHOT</version>
|
||||
|
||||
<name>ratelimiter</name>
|
||||
<url>https://reflectoring.io</url>
|
||||
|
||||
<properties>
|
||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||
<!-- <maven.compiler.source>1.8</maven.compiler.source>-->
|
||||
<!-- <maven.compiler.target>1.8</maven.compiler.target>-->
|
||||
<maven.compiler.release>11</maven.compiler.release>
|
||||
</properties>
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>junit</groupId>
|
||||
<artifactId>junit</artifactId>
|
||||
<version>4.11</version>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
|
||||
<!-- https://mvnrepository.com/artifact/io.github.resilience4j/resilience4j-ratelimiter -->
|
||||
<dependency>
|
||||
<groupId>io.github.resilience4j</groupId>
|
||||
<artifactId>resilience4j-ratelimiter</artifactId>
|
||||
<version>1.5.0</version>
|
||||
</dependency>
|
||||
|
||||
<!-- https://mvnrepository.com/artifact/io.github.resilience4j/resilience4j-core -->
|
||||
<dependency>
|
||||
<groupId>io.github.resilience4j</groupId>
|
||||
<artifactId>resilience4j-core</artifactId>
|
||||
<version>1.5.0</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>io.github.resilience4j</groupId>
|
||||
<artifactId>resilience4j-all</artifactId>
|
||||
<version>1.5.0</version>
|
||||
</dependency>
|
||||
|
||||
<dependency>
|
||||
<groupId>org.projectlombok</groupId>
|
||||
<artifactId>lombok</artifactId>
|
||||
<version>1.18.12</version>
|
||||
</dependency>
|
||||
|
||||
<!-- https://mvnrepository.com/artifact/io.github.resilience4j/resilience4j-micrometer -->
|
||||
<dependency>
|
||||
<groupId>io.github.resilience4j</groupId>
|
||||
<artifactId>resilience4j-micrometer</artifactId>
|
||||
<version>1.5.0</version>
|
||||
</dependency>
|
||||
|
||||
<!-- https://mvnrepository.com/artifact/io.micrometer/micrometer-core -->
|
||||
<dependency>
|
||||
<groupId>io.micrometer</groupId>
|
||||
<artifactId>micrometer-core</artifactId>
|
||||
<version>1.5.0</version>
|
||||
</dependency>
|
||||
|
||||
</dependencies>
|
||||
|
||||
<build>
|
||||
<pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) -->
|
||||
<plugins>
|
||||
<!-- clean lifecycle, see https://maven.apache.org/ref/current/maven-core/lifecycles.html#clean_Lifecycle -->
|
||||
<plugin>
|
||||
<artifactId>maven-clean-plugin</artifactId>
|
||||
<version>3.1.0</version>
|
||||
</plugin>
|
||||
<!-- default lifecycle, jar packaging: see https://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_jar_packaging -->
|
||||
<plugin>
|
||||
<artifactId>maven-resources-plugin</artifactId>
|
||||
<version>3.0.2</version>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-compiler-plugin</artifactId>
|
||||
<version>3.8.0</version>
|
||||
<configuration>
|
||||
<release>11</release>
|
||||
</configuration>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-surefire-plugin</artifactId>
|
||||
<version>2.22.1</version>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-jar-plugin</artifactId>
|
||||
<version>3.0.2</version>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-install-plugin</artifactId>
|
||||
<version>2.5.2</version>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-deploy-plugin</artifactId>
|
||||
<version>2.8.2</version>
|
||||
</plugin>
|
||||
<!-- site lifecycle, see https://maven.apache.org/ref/current/maven-core/lifecycles.html#site_Lifecycle -->
|
||||
<plugin>
|
||||
<artifactId>maven-site-plugin</artifactId>
|
||||
<version>3.7.1</version>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<artifactId>maven-project-info-reports-plugin</artifactId>
|
||||
<version>3.0.0</version>
|
||||
</plugin>
|
||||
</plugins>
|
||||
</pluginManagement>
|
||||
</build>
|
||||
</project>
|
||||
@@ -0,0 +1,312 @@
|
||||
package io.reflectoring.resilience4j.ratelimit;
|
||||
|
||||
import io.github.resilience4j.micrometer.tagged.TaggedRateLimiterMetrics;
|
||||
import io.github.resilience4j.ratelimiter.RateLimiter;
|
||||
import io.github.resilience4j.ratelimiter.RateLimiterConfig;
|
||||
import io.github.resilience4j.ratelimiter.RateLimiterRegistry;
|
||||
import io.github.resilience4j.retry.Retry;
|
||||
import io.github.resilience4j.retry.RetryConfig;
|
||||
import io.github.resilience4j.retry.RetryRegistry;
|
||||
import io.micrometer.core.instrument.Meter;
|
||||
import io.micrometer.core.instrument.MeterRegistry;
|
||||
import io.micrometer.core.instrument.simple.SimpleMeterRegistry;
|
||||
import io.reflectoring.resilience4j.ratelimit.model.Flight;
|
||||
import io.reflectoring.resilience4j.ratelimit.model.SearchRequest;
|
||||
import io.reflectoring.resilience4j.ratelimit.services.FlightSearchService;
|
||||
import io.vavr.CheckedFunction0;
|
||||
import java.time.Duration;
|
||||
import java.util.List;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.function.Consumer;
|
||||
import java.util.function.Supplier;
|
||||
import java.util.stream.StreamSupport;
|
||||
|
||||
public class Examples
|
||||
{
|
||||
void displayDefaultValues() {
|
||||
RateLimiterConfig config = RateLimiterConfig.ofDefaults();
|
||||
|
||||
System.out.println("Limit for period = " + config.getLimitForPeriod());
|
||||
System.out.println(("Refresh period = " + Duration.from(config.getLimitRefreshPeriod()).toNanos()));
|
||||
System.out.println("Timeout value = " + Duration.from(config.getTimeoutDuration()).toMillis());
|
||||
}
|
||||
|
||||
void basicExample() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofSeconds(1)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
for (int i=0; i<3; i++) {
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
}
|
||||
|
||||
void timeoutExample() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofMillis(250)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
for (int i=0; i<3; i++) {
|
||||
try {
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void checkedExceptionExample() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofSeconds(1)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
CheckedFunction0<List<Flight>> flights = RateLimiter.decorateCheckedSupplier(limiter, () -> service.searchFlightsThrowingException(request));
|
||||
try {
|
||||
System.out.println(flights.apply());
|
||||
} catch (Throwable throwable) {
|
||||
throwable.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
void usagePattern() {
|
||||
RateLimiterConfig config = RateLimiterConfig.ofDefaults();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
|
||||
|
||||
void multipleLimits_2rps_40rpm_sequential() {
|
||||
RateLimiterConfig rpsConfig = RateLimiterConfig.custom().
|
||||
limitForPeriod(2).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofMillis(2000)).build();
|
||||
RateLimiterConfig rpmConfig = RateLimiterConfig.custom().
|
||||
limitForPeriod(40).
|
||||
limitRefreshPeriod(Duration.ofMinutes(1)).
|
||||
timeoutDuration(Duration.ofMillis(2000)).build();
|
||||
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(rpsConfig);
|
||||
RateLimiter rpsLimiter = registry.rateLimiter("flightSearchService_rps", rpsConfig);
|
||||
RateLimiter rpmLimiter = registry.rateLimiter("flightSearchService_rpm", rpmConfig);
|
||||
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> rpsLimitedSupplier = RateLimiter.decorateSupplier(rpsLimiter, () -> service.searchFlights(request));
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(rpmLimiter, rpsLimitedSupplier);
|
||||
for (int i=0; i<45; i++) {
|
||||
try {
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void changeLimitsExample() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofSeconds(1)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
for (int i=0; i<3; i++) {
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
|
||||
limiter.changeLimitForPeriod(2);
|
||||
limiter.changeTimeoutDuration(Duration.ofSeconds(2));
|
||||
|
||||
for (int i=0; i<4; i++) {
|
||||
System.out.println(flightsSupplier.get());
|
||||
}
|
||||
}
|
||||
|
||||
void retryAndRateLimit() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofMillis(250)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter rateLimiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
|
||||
RetryConfig retryConfig = RetryConfig.custom().maxAttempts(2).waitDuration(Duration.ofSeconds(1)).build();
|
||||
RetryRegistry retryRegistry = RetryRegistry.of(retryConfig);
|
||||
Retry retry = retryRegistry.retry("rateLimitedFlightSearch", retryConfig);
|
||||
retry.getEventPublisher().onRetry(e -> System.out.println(e.toString()));
|
||||
retry.getEventPublisher().onSuccess(e -> System.out.println(e.toString()));
|
||||
|
||||
Supplier<List<Flight>> rateLimitedFlightsSupplier = RateLimiter.decorateSupplier(rateLimiter, () -> service.searchFlights(request));
|
||||
Supplier<List<Flight>> retryingFlightsSupplier = Retry.decorateSupplier(retry, rateLimitedFlightsSupplier);
|
||||
|
||||
System.out.println(retryingFlightsSupplier.get());
|
||||
System.out.println(retryingFlightsSupplier.get());
|
||||
}
|
||||
|
||||
private void limit_1rps_concurrent() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofMillis(50)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
|
||||
CompletableFuture.supplyAsync(flightsSupplier).thenAccept(System.out::println);
|
||||
CompletableFuture.supplyAsync(flightsSupplier).whenComplete((flights, e) -> {
|
||||
if (e != null) e.printStackTrace();
|
||||
}).thenAccept(System.out::println);
|
||||
try {
|
||||
Thread.sleep(3000);
|
||||
} catch (InterruptedException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
void rateLimiterEvents() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofMillis(50)).build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter limiter = registry.rateLimiter("flightSearchService");
|
||||
|
||||
limiter.getEventPublisher().onSuccess(e -> System.out.println(e.toString()));
|
||||
limiter.getEventPublisher().onFailure(e -> System.out.println(e.toString()));
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(limiter, () -> service.searchFlights(request));
|
||||
try {
|
||||
flightsSupplier.get();
|
||||
flightsSupplier.get();
|
||||
}
|
||||
catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
void rateLimiterMetrics() {
|
||||
RateLimiterConfig config = RateLimiterConfig.custom().
|
||||
limitForPeriod(1).
|
||||
limitRefreshPeriod(Duration.ofSeconds(1)).
|
||||
timeoutDuration(Duration.ofSeconds(1)).
|
||||
build();
|
||||
RateLimiterRegistry registry = RateLimiterRegistry.of(config);
|
||||
RateLimiter retry = registry.rateLimiter("flightSearchService", config);
|
||||
|
||||
MeterRegistry meterRegistry = new SimpleMeterRegistry();
|
||||
TaggedRateLimiterMetrics.ofRateLimiterRegistry(registry).bindTo(meterRegistry);
|
||||
|
||||
FlightSearchService service = new FlightSearchService();
|
||||
|
||||
SearchRequest request = new SearchRequest("NYC", "LAX", "07/31/2020");
|
||||
Supplier<List<Flight>> flightsSupplier = RateLimiter.decorateSupplier(retry, () -> service.searchFlights(request));
|
||||
|
||||
int count = 0;
|
||||
for (int i=0; i<50; i++) {
|
||||
count++;
|
||||
CompletableFuture.supplyAsync(flightsSupplier).whenComplete((flights, e) -> {
|
||||
if (e != null) e.printStackTrace();
|
||||
}).thenAccept(System.out::println);
|
||||
|
||||
if (count % 10 == 0) {
|
||||
printMetricDetails(meterRegistry);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void printMetricDetails(MeterRegistry meterRegistry) {
|
||||
Consumer<Meter> meterConsumer = meter -> {
|
||||
String desc = meter.getId().getDescription();
|
||||
String metricName = meter.getId().getName();
|
||||
Double metricValue = StreamSupport.stream(meter.measure().spliterator(), false).
|
||||
filter(m -> m.getStatistic().name().equals("VALUE")).
|
||||
findFirst().
|
||||
map(m -> m.getValue()).
|
||||
orElse(0.0);
|
||||
System.out.println(desc + " - " + metricName + ": " + metricValue);
|
||||
};
|
||||
meterRegistry.forEachMeter(meterConsumer);
|
||||
}
|
||||
|
||||
public static void main( String[] args ) {
|
||||
Examples examples = new Examples();
|
||||
System.out.println("---------------------------- displayDefaultValues -------------------------------------------");
|
||||
examples.displayDefaultValues();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("----------------------------- usagePattern ------------------------------------------");
|
||||
examples.usagePattern();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("----------------------------- basicExample ------------------------------------------");
|
||||
examples.basicExample();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("----------------------------- timeoutExample ------------------------------------------");
|
||||
examples.timeoutExample();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------ checkedExceptionExample -----------------------------------------");
|
||||
examples.checkedExceptionExample();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------ multipleLimits_2rps_40rpm_sequential -----------------------------------------");
|
||||
examples.multipleLimits_2rps_40rpm_sequential();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------- changeLimitsExample ----------------------------------------");
|
||||
examples.changeLimitsExample();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------- retryAndRateLimit ----------------------------------------");
|
||||
examples.retryAndRateLimit();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------- limit_1rps_concurrent ----------------------------------------");
|
||||
examples.limit_1rps_concurrent();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------ rateLimiterEvents -----------------------------------------");
|
||||
examples.rateLimiterEvents();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
System.out.println("------------------------------- rateLimiterMetrics ----------------------------------------");
|
||||
examples.rateLimiterMetrics();
|
||||
System.out.println("-----------------------------------------------------------------------");
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,60 @@
|
||||
package io.reflectoring.resilience4j.ratelimit.model;
|
||||
|
||||
public class Flight {
|
||||
String flightNumber;
|
||||
String flightDate;
|
||||
String from;
|
||||
String to;
|
||||
|
||||
public Flight() {
|
||||
}
|
||||
|
||||
public Flight(String flightNumber, String flightDate, String from, String to) {
|
||||
this.flightNumber = flightNumber;
|
||||
this.flightDate = flightDate;
|
||||
this.from = from;
|
||||
this.to = to;
|
||||
}
|
||||
|
||||
public String getFlightNumber() {
|
||||
return flightNumber;
|
||||
}
|
||||
|
||||
public void setFlightNumber(String flightNumber) {
|
||||
this.flightNumber = flightNumber;
|
||||
}
|
||||
|
||||
public String getFlightDate() {
|
||||
return flightDate;
|
||||
}
|
||||
|
||||
public void setFlightDate(String flightDate) {
|
||||
this.flightDate = flightDate;
|
||||
}
|
||||
|
||||
public String getFrom() {
|
||||
return from;
|
||||
}
|
||||
|
||||
public void setFrom(String from) {
|
||||
this.from = from;
|
||||
}
|
||||
|
||||
public String getTo() {
|
||||
return to;
|
||||
}
|
||||
|
||||
public void setTo(String to) {
|
||||
this.to = to;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "Flight{" +
|
||||
"flightNumber='" + flightNumber + '\'' +
|
||||
", flightDate='" + flightDate + '\'' +
|
||||
", from='" + from + '\'' +
|
||||
", to='" + to + '\'' +
|
||||
'}';
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,25 @@
|
||||
package io.reflectoring.resilience4j.ratelimit.model;
|
||||
|
||||
public class SearchRequest {
|
||||
String from;
|
||||
String to;
|
||||
String flightDate;
|
||||
|
||||
public SearchRequest(String from, String to, String flightDate) {
|
||||
this.from = from;
|
||||
this.to = to;
|
||||
this.flightDate = flightDate;
|
||||
}
|
||||
|
||||
public String getFrom() {
|
||||
return from;
|
||||
}
|
||||
|
||||
public String getTo() {
|
||||
return to;
|
||||
}
|
||||
|
||||
public String getFlightDate() {
|
||||
return flightDate;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,34 @@
|
||||
package io.reflectoring.resilience4j.ratelimit.services;
|
||||
|
||||
import io.reflectoring.resilience4j.ratelimit.model.Flight;
|
||||
import io.reflectoring.resilience4j.ratelimit.model.SearchRequest;
|
||||
import java.time.LocalDateTime;
|
||||
import java.time.format.DateTimeFormatter;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
public class FlightSearchService {
|
||||
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("HH:mm:ss SSS");
|
||||
|
||||
public List<Flight> searchFlights(SearchRequest request) {
|
||||
System.out.println("Searching for flights; "
|
||||
+ "current time = " + LocalDateTime.now().format(formatter) +
|
||||
"; current thread = " + Thread.currentThread().getName());
|
||||
|
||||
List<Flight> flights = Arrays.asList(
|
||||
new Flight("XY 765", request.getFlightDate(), request.getFrom(), request.getTo()),
|
||||
new Flight("XY 746", request.getFlightDate(), request.getFrom(), request.getTo())
|
||||
);
|
||||
System.out.println("Flight search successful");
|
||||
return flights;
|
||||
}
|
||||
|
||||
public List<Flight> searchFlightsThrowingException(SearchRequest request) throws Exception {
|
||||
System.out.println("Searching for flights; "
|
||||
+ "current time = " + LocalDateTime.now().format(formatter) +
|
||||
"; current thread = " + Thread.currentThread().getName());
|
||||
|
||||
throw new Exception("Exception when searching for flights");
|
||||
}
|
||||
|
||||
}
|
||||
7
resilience4j/retry/README.md
Normal file
7
resilience4j/retry/README.md
Normal file
@@ -0,0 +1,7 @@
|
||||
# Retry with Resilience4J
|
||||
|
||||
Run the Examples program
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [Implementing Retry with Resilience4j](https://reflectoring.io/retry-with-resilience4j/)
|
||||
@@ -1 +0,0 @@
|
||||
Run the Examples program
|
||||
@@ -9,7 +9,7 @@
|
||||
<version>1.0-SNAPSHOT</version>
|
||||
|
||||
<name>retry</name>
|
||||
<url>http://www.reflectoring.io</url>
|
||||
<url>https://reflectoring.io</url>
|
||||
|
||||
<properties>
|
||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||
|
||||
7
solid/isp/README.md
Normal file
7
solid/isp/README.md
Normal file
@@ -0,0 +1,7 @@
|
||||
# Interface Segregation Principle
|
||||
|
||||
Example code showing the Interface Segregation Principle
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [Everything You Need to Know About the Interface Segregation Principle](https://reflectoring.io/interface-segregation-principle/)
|
||||
0
solid/gradlew → solid/isp/gradlew
vendored
0
solid/gradlew → solid/isp/gradlew
vendored
@@ -1,3 +1,5 @@
|
||||
# Liskov Substition Principle
|
||||
|
||||
There are 5 programs under com/reflectoring/examples/lsp/paymentexample.
|
||||
|
||||
Run in this order to see class design evolution:
|
||||
@@ -11,4 +13,8 @@ Packages are organized similarly: violation, forcefit, forcefitandconditional,
|
||||
and redesigned have specific classes modified at that point in the evolution.
|
||||
|
||||
Common code is in common, common/exceptions, common/external and
|
||||
common/instruments.
|
||||
common/instruments.
|
||||
|
||||
## Blog posts
|
||||
|
||||
* [The Liskov Substitution Principle Explained](https://reflectoring.io/lsp-explained/)
|
||||
26
spring-boot/bean-lifecycle/.gitignore
vendored
Normal file
26
spring-boot/bean-lifecycle/.gitignore
vendored
Normal file
@@ -0,0 +1,26 @@
|
||||
.gradle
|
||||
/build/
|
||||
!gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
### STS ###
|
||||
.apt_generated
|
||||
.classpath
|
||||
.factorypath
|
||||
.project
|
||||
.settings
|
||||
.springBeans
|
||||
.sts4-cache
|
||||
|
||||
### IntelliJ IDEA ###
|
||||
.idea
|
||||
*.iws
|
||||
*.iml
|
||||
*.ipr
|
||||
/out/
|
||||
|
||||
### NetBeans ###
|
||||
/nbproject/private/
|
||||
/nbbuild/
|
||||
/dist/
|
||||
/nbdist/
|
||||
/.nb-gradle/
|
||||
5
spring-boot/bean-lifecycle/README.md
Normal file
5
spring-boot/bean-lifecycle/README.md
Normal file
@@ -0,0 +1,5 @@
|
||||
# Spring Bean Lifecycle
|
||||
|
||||
This code shows how to hook in to the phases of the Spring bean lifecycle.
|
||||
|
||||
Read more in the Blog post: [Hooking into the Spring Bean Lifecycle](https://reflectoring.io/spring-bean-lifecycle)
|
||||
26
spring-boot/bean-lifecycle/build.gradle
Normal file
26
spring-boot/bean-lifecycle/build.gradle
Normal file
@@ -0,0 +1,26 @@
|
||||
plugins {
|
||||
id 'org.springframework.boot' version '2.3.2.RELEASE'
|
||||
id 'io.spring.dependency-management' version '1.0.9.RELEASE'
|
||||
id 'java'
|
||||
}
|
||||
|
||||
group = 'io.reflectoring'
|
||||
version = '0.0.1-SNAPSHOT'
|
||||
sourceCompatibility = '11'
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
implementation 'org.springframework.boot:spring-boot-starter'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-quartz'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-jersey'
|
||||
testImplementation('org.springframework.boot:spring-boot-starter-test') {
|
||||
exclude group: 'org.junit.vintage', module: 'junit-vintage-engine'
|
||||
}
|
||||
}
|
||||
|
||||
test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
BIN
spring-boot/bean-lifecycle/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
spring-boot/bean-lifecycle/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
spring-boot/bean-lifecycle/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
spring-boot/bean-lifecycle/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@@ -0,0 +1,6 @@
|
||||
#Tue Feb 06 12:27:20 CET 2018
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-bin.zip
|
||||
172
spring-boot/bean-lifecycle/gradlew
vendored
Executable file
172
spring-boot/bean-lifecycle/gradlew
vendored
Executable file
@@ -0,0 +1,172 @@
|
||||
#!/usr/bin/env sh
|
||||
|
||||
##############################################################################
|
||||
##
|
||||
## Gradle start up script for UN*X
|
||||
##
|
||||
##############################################################################
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
# Resolve links: $0 may be a link
|
||||
PRG="$0"
|
||||
# Need this for relative symlinks.
|
||||
while [ -h "$PRG" ] ; do
|
||||
ls=`ls -ld "$PRG"`
|
||||
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||
if expr "$link" : '/.*' > /dev/null; then
|
||||
PRG="$link"
|
||||
else
|
||||
PRG=`dirname "$PRG"`"/$link"
|
||||
fi
|
||||
done
|
||||
SAVED="`pwd`"
|
||||
cd "`dirname \"$PRG\"`/" >/dev/null
|
||||
APP_HOME="`pwd -P`"
|
||||
cd "$SAVED" >/dev/null
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=`basename "$0"`
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS=""
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD="maximum"
|
||||
|
||||
warn ( ) {
|
||||
echo "$*"
|
||||
}
|
||||
|
||||
die ( ) {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
}
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
nonstop=false
|
||||
case "`uname`" in
|
||||
CYGWIN* )
|
||||
cygwin=true
|
||||
;;
|
||||
Darwin* )
|
||||
darwin=true
|
||||
;;
|
||||
MINGW* )
|
||||
msys=true
|
||||
;;
|
||||
NONSTOP* )
|
||||
nonstop=true
|
||||
;;
|
||||
esac
|
||||
|
||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||
else
|
||||
JAVACMD="$JAVA_HOME/bin/java"
|
||||
fi
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
else
|
||||
JAVACMD="java"
|
||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
|
||||
MAX_FD_LIMIT=`ulimit -H -n`
|
||||
if [ $? -eq 0 ] ; then
|
||||
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||
MAX_FD="$MAX_FD_LIMIT"
|
||||
fi
|
||||
ulimit -n $MAX_FD
|
||||
if [ $? -ne 0 ] ; then
|
||||
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||
fi
|
||||
else
|
||||
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||
fi
|
||||
fi
|
||||
|
||||
# For Darwin, add options to specify how the application appears in the dock
|
||||
if $darwin; then
|
||||
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||
fi
|
||||
|
||||
# For Cygwin, switch paths to Windows format before running java
|
||||
if $cygwin ; then
|
||||
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||
|
||||
# We build the pattern for arguments to be converted via cygpath
|
||||
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||
SEP=""
|
||||
for dir in $ROOTDIRSRAW ; do
|
||||
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||
SEP="|"
|
||||
done
|
||||
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||
# Add a user-defined pattern to the cygpath arguments
|
||||
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||
fi
|
||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||
i=0
|
||||
for arg in "$@" ; do
|
||||
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||
|
||||
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||
else
|
||||
eval `echo args$i`="\"$arg\""
|
||||
fi
|
||||
i=$((i+1))
|
||||
done
|
||||
case $i in
|
||||
(0) set -- ;;
|
||||
(1) set -- "$args0" ;;
|
||||
(2) set -- "$args0" "$args1" ;;
|
||||
(3) set -- "$args0" "$args1" "$args2" ;;
|
||||
(4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||
(5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||
(6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||
(7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||
(8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||
(9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||
esac
|
||||
fi
|
||||
|
||||
# Escape application args
|
||||
save ( ) {
|
||||
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||
echo " "
|
||||
}
|
||||
APP_ARGS=$(save "$@")
|
||||
|
||||
# Collect all arguments for the java command, following the shell quoting and substitution rules
|
||||
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
|
||||
|
||||
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
|
||||
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
|
||||
cd "$(dirname "$0")"
|
||||
fi
|
||||
|
||||
exec "$JAVACMD" "$@"
|
||||
84
spring-boot/bean-lifecycle/gradlew.bat
vendored
Normal file
84
spring-boot/bean-lifecycle/gradlew.bat
vendored
Normal file
@@ -0,0 +1,84 @@
|
||||
@if "%DEBUG%" == "" @echo off
|
||||
@rem ##########################################################################
|
||||
@rem
|
||||
@rem Gradle startup script for Windows
|
||||
@rem
|
||||
@rem ##########################################################################
|
||||
|
||||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%" == "" set DIRNAME=.
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS=
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
set JAVA_EXE=java.exe
|
||||
%JAVA_EXE% -version >NUL 2>&1
|
||||
if "%ERRORLEVEL%" == "0" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:findJavaFromJavaHome
|
||||
set JAVA_HOME=%JAVA_HOME:"=%
|
||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||
|
||||
if exist "%JAVA_EXE%" goto init
|
||||
|
||||
echo.
|
||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||
echo.
|
||||
echo Please set the JAVA_HOME variable in your environment to match the
|
||||
echo location of your Java installation.
|
||||
|
||||
goto fail
|
||||
|
||||
:init
|
||||
@rem Get command-line arguments, handling Windows variants
|
||||
|
||||
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||
|
||||
:win9xME_args
|
||||
@rem Slurp the command line arguments.
|
||||
set CMD_LINE_ARGS=
|
||||
set _SKIP=2
|
||||
|
||||
:win9xME_args_slurp
|
||||
if "x%~1" == "x" goto execute
|
||||
|
||||
set CMD_LINE_ARGS=%*
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||
|
||||
@rem Execute Gradle
|
||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS%
|
||||
|
||||
:end
|
||||
@rem End local scope for the variables with windows NT shell
|
||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||
|
||||
:fail
|
||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||
rem the _cmd.exe /c_ return code!
|
||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||
exit /b 1
|
||||
|
||||
:mainEnd
|
||||
if "%OS%"=="Windows_NT" endlocal
|
||||
|
||||
:omega
|
||||
1
spring-boot/bean-lifecycle/settings.gradle
Normal file
1
spring-boot/bean-lifecycle/settings.gradle
Normal file
@@ -0,0 +1 @@
|
||||
rootProject.name = 'beanlifecycle'
|
||||
@@ -0,0 +1,12 @@
|
||||
package io.reflectoring.beanlifecycle;
|
||||
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
|
||||
@SpringBootApplication
|
||||
public class BeanLifecycleApplication {
|
||||
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(BeanLifecycleApplication.class, args);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,26 @@
|
||||
package io.reflectoring.beanlifecycle;
|
||||
|
||||
import org.springframework.beans.BeansException;
|
||||
import org.springframework.beans.factory.config.BeanPostProcessor;
|
||||
|
||||
public class MyBeanPostProcessor implements BeanPostProcessor {
|
||||
|
||||
@Override
|
||||
public Object postProcessBeforeInitialization(Object bean, String beanName)
|
||||
throws BeansException {
|
||||
if (bean instanceof MySpringBean) {
|
||||
System.out.println("--- postProcessBeforeInitialization executed ---");
|
||||
}
|
||||
return bean;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object postProcessAfterInitialization(Object bean, String beanName)
|
||||
throws BeansException {
|
||||
if (bean instanceof MySpringBean) {
|
||||
System.out.println("--- postProcessAfterInitialization executed ---");
|
||||
}
|
||||
return bean;
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,65 @@
|
||||
package io.reflectoring.beanlifecycle;
|
||||
|
||||
import org.springframework.beans.BeansException;
|
||||
import org.springframework.beans.factory.BeanNameAware;
|
||||
import org.springframework.beans.factory.DisposableBean;
|
||||
import org.springframework.beans.factory.InitializingBean;
|
||||
import org.springframework.context.ApplicationContext;
|
||||
import org.springframework.context.ApplicationContextAware;
|
||||
|
||||
import javax.annotation.PostConstruct;
|
||||
import javax.annotation.PreDestroy;
|
||||
|
||||
public class MySpringBean implements BeanNameAware, ApplicationContextAware,
|
||||
InitializingBean, DisposableBean {
|
||||
|
||||
private String message;
|
||||
|
||||
public void sendMessage(String message) {
|
||||
this.message = message;
|
||||
}
|
||||
|
||||
public String getMessage() {
|
||||
return this.message;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setBeanName(String name) {
|
||||
System.out.println("--- setBeanName executed ---");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setApplicationContext(ApplicationContext applicationContext)
|
||||
throws BeansException {
|
||||
System.out.println("--- setApplicationContext executed ---");
|
||||
}
|
||||
|
||||
@PostConstruct
|
||||
public void postConstruct() {
|
||||
System.out.println("--- @PostConstruct executed ---");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void afterPropertiesSet() {
|
||||
System.out.println("--- afterPropertiesSet executed ---");
|
||||
}
|
||||
|
||||
public void initMethod() {
|
||||
System.out.println("--- init-method executed ---");
|
||||
}
|
||||
|
||||
@PreDestroy
|
||||
public void preDestroy() {
|
||||
System.out.println("--- @PreDestroy executed ---");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void destroy() throws Exception {
|
||||
System.out.println("--- destroy executed ---");
|
||||
}
|
||||
|
||||
public void destroyMethod() {
|
||||
System.out.println("--- destroy-method executed ---");
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,19 @@
|
||||
package io.reflectoring.beanlifecycle;
|
||||
|
||||
import org.springframework.context.annotation.Bean;
|
||||
import org.springframework.context.annotation.Configuration;
|
||||
|
||||
@Configuration
|
||||
public class MySpringConfiguration {
|
||||
|
||||
@Bean
|
||||
public MyBeanPostProcessor myBeanPostProcessor(){
|
||||
return new MyBeanPostProcessor();
|
||||
}
|
||||
|
||||
@Bean(initMethod = "initMethod", destroyMethod = "destroyMethod")
|
||||
public MySpringBean mySpringBean(){
|
||||
return new MySpringBean();
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,16 @@
|
||||
package io.reflectoring.beanlifecycle.ipdatabase;
|
||||
|
||||
public class IpDatabaseRepository {
|
||||
|
||||
private String file;
|
||||
|
||||
public IpDatabaseRepository(String file) {
|
||||
this.file = file;
|
||||
// some logic to download and update the new database file
|
||||
}
|
||||
|
||||
public String lookup(String ipAddress){
|
||||
// hardcoded for examplary purposes
|
||||
return "DE";
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,38 @@
|
||||
package io.reflectoring.beanlifecycle.ipdatabase;
|
||||
|
||||
import org.springframework.beans.BeansException;
|
||||
import org.springframework.beans.factory.BeanFactory;
|
||||
import org.springframework.beans.factory.BeanFactoryAware;
|
||||
import org.springframework.beans.factory.support.AbstractBeanDefinition;
|
||||
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
|
||||
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
|
||||
import org.springframework.stereotype.Service;
|
||||
|
||||
@Service
|
||||
public class IpToLocationService implements BeanFactoryAware {
|
||||
|
||||
DefaultListableBeanFactory listableBeanFactory;
|
||||
IpDatabaseRepository ipDatabaseRepository;
|
||||
|
||||
@Override
|
||||
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
|
||||
listableBeanFactory = (DefaultListableBeanFactory) beanFactory;
|
||||
updateIpDatabase();
|
||||
}
|
||||
|
||||
public void updateIpDatabase(){
|
||||
String updateUrl = "https://download.acme.com/latest/ip-address-database.mdb";
|
||||
|
||||
AbstractBeanDefinition definition = BeanDefinitionBuilder
|
||||
.genericBeanDefinition(IpDatabaseRepository.class)
|
||||
.addConstructorArgValue(updateUrl)
|
||||
.getBeanDefinition();
|
||||
|
||||
listableBeanFactory.registerBeanDefinition("ipDatabaseRepository", definition);
|
||||
this.ipDatabaseRepository = listableBeanFactory.getBean(IpDatabaseRepository.class);
|
||||
}
|
||||
|
||||
public String getCountry(String ipAddress){
|
||||
return ipDatabaseRepository.lookup(ipAddress);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,22 @@
|
||||
package io.reflectoring.beanlifecycle.jersey;
|
||||
|
||||
import org.glassfish.jersey.server.ResourceConfig;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.context.ApplicationContext;
|
||||
import org.springframework.context.annotation.Configuration;
|
||||
|
||||
import javax.annotation.PostConstruct;
|
||||
import javax.ws.rs.Path;
|
||||
|
||||
@Configuration
|
||||
public class JerseyConfig extends ResourceConfig {
|
||||
|
||||
@Autowired
|
||||
private ApplicationContext applicationContext;
|
||||
|
||||
@PostConstruct
|
||||
public void registerResources() {
|
||||
applicationContext.getBeansWithAnnotation(Path.class).values().forEach(this::register);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,34 @@
|
||||
package io.reflectoring.beanlifecycle.jersey;
|
||||
|
||||
import io.reflectoring.beanlifecycle.ipdatabase.IpToLocationService;
|
||||
import org.springframework.beans.factory.annotation.Autowired;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
import javax.servlet.http.HttpServletRequest;
|
||||
import javax.ws.rs.GET;
|
||||
import javax.ws.rs.POST;
|
||||
import javax.ws.rs.Path;
|
||||
|
||||
@Component
|
||||
@Path("/api")
|
||||
public class LocationResource {
|
||||
|
||||
@Autowired
|
||||
HttpServletRequest servletRequest;
|
||||
|
||||
@Autowired
|
||||
IpToLocationService locationService;
|
||||
|
||||
@GET
|
||||
@Path("/location")
|
||||
public String location() {
|
||||
return locationService.getCountry(servletRequest.getRemoteAddr());
|
||||
}
|
||||
|
||||
@POST
|
||||
@Path("/update")
|
||||
public void update() {
|
||||
locationService.updateIpDatabase();
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
package io.reflectoring.beanlifecycle.logging;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
import org.springframework.beans.factory.BeanNameAware;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
@Component
|
||||
public class NamedSpringBean implements BeanNameAware {
|
||||
|
||||
Logger logger = LoggerFactory.getLogger(NamedSpringBean.class);
|
||||
|
||||
public void setBeanName(String name) {
|
||||
logger.info(name + " created.");
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,28 @@
|
||||
package io.reflectoring.beanlifecycle.quartz;
|
||||
|
||||
import org.quartz.spi.TriggerFiredBundle;
|
||||
import org.springframework.beans.factory.config.AutowireCapableBeanFactory;
|
||||
import org.springframework.context.ApplicationContext;
|
||||
import org.springframework.context.ApplicationContextAware;
|
||||
import org.springframework.scheduling.quartz.SpringBeanJobFactory;
|
||||
|
||||
/**
|
||||
* Custom Quartz {@link org.quartz.spi.JobFactory} to provide bean autowire capability
|
||||
*/
|
||||
public class AutowireCapableJobFactory extends SpringBeanJobFactory implements ApplicationContextAware {
|
||||
|
||||
private AutowireCapableBeanFactory beanFactory;
|
||||
|
||||
@Override
|
||||
public void setApplicationContext(final ApplicationContext context) {
|
||||
this.beanFactory = context.getAutowireCapableBeanFactory();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Object createJobInstance(final TriggerFiredBundle bundle) throws Exception {
|
||||
final Object job = super.createJobInstance(bundle);
|
||||
this.beanFactory.autowireBean(job);
|
||||
return job;
|
||||
}
|
||||
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user