Compare commits
417 Commits
reflect-92
...
dependabot
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5b4636805e | ||
|
|
dbdd4d20cb | ||
|
|
379f0d6799 | ||
|
|
8e299cf819 | ||
|
|
14536868c5 | ||
|
|
bbca460ed2 | ||
|
|
cd2e56a959 | ||
|
|
143a5b53d0 | ||
|
|
fec088b7db | ||
|
|
d68609b407 | ||
|
|
8b1ae3f3ed | ||
|
|
0ccd671636 | ||
|
|
1dac3ad175 | ||
|
|
91e38a8361 | ||
|
|
071651aa67 | ||
|
|
a6c5ab4cd7 | ||
|
|
c28b2d4a73 | ||
|
|
8bca7692e2 | ||
|
|
d1b5b19dd6 | ||
|
|
d5ff8dae58 | ||
|
|
4bb5bb8af9 | ||
|
|
e03595d5bd | ||
|
|
b5cfe4ad2a | ||
|
|
4a64888977 | ||
|
|
44d082c163 | ||
|
|
8b3fc4cf5c | ||
|
|
5bd780f9c8 | ||
|
|
eba824b908 | ||
|
|
381be70991 | ||
|
|
87a6cac6b3 | ||
|
|
fef6c48fe1 | ||
|
|
3f59bd2ea4 | ||
|
|
e2d6cc3acb | ||
|
|
fef53fa72a | ||
|
|
6b2ad23059 | ||
|
|
368a059cfe | ||
|
|
67b4cbe08f | ||
|
|
0514f9f517 | ||
|
|
76bcc6c016 | ||
|
|
fd14c4c13e | ||
|
|
39dbdc129f | ||
|
|
dbaa38a5fe | ||
|
|
8f6e222ef2 | ||
|
|
88c5ff3dc7 | ||
|
|
4cf2cbc10c | ||
|
|
e4e3d8a55b | ||
|
|
d41fa4a006 | ||
|
|
48cdce6e65 | ||
|
|
2d8c099993 | ||
|
|
1c92604f3b | ||
|
|
6a63059b8c | ||
|
|
1de287bcdb | ||
|
|
f044fcc228 | ||
|
|
ee1712c245 | ||
|
|
8f03b89816 | ||
|
|
f751203867 | ||
|
|
4c4469b43b | ||
|
|
0e40dbfee3 | ||
|
|
d47db42925 | ||
|
|
04dcc8097a | ||
|
|
ec52966e5f | ||
|
|
b982958196 | ||
|
|
974872ed8e | ||
|
|
0edc9e3e8e | ||
|
|
7d4dd46e2c | ||
|
|
5169b74c86 | ||
|
|
b5dc195722 | ||
|
|
47a1d0c4b7 | ||
|
|
cacef37325 | ||
|
|
ab9aa60613 | ||
|
|
782b64a32f | ||
|
|
2f948c0616 | ||
|
|
2e3d9a03b6 | ||
|
|
0e48de379f | ||
|
|
08d2592311 | ||
|
|
313889e494 | ||
|
|
6058641b80 | ||
|
|
ba8ba61586 | ||
|
|
ec595e8bae | ||
|
|
743bf01b0d | ||
|
|
ca9cc4ad65 | ||
|
|
b2ee3aafa3 | ||
|
|
f137ea52db | ||
|
|
13888f806e | ||
|
|
778a3d1d6f | ||
|
|
8b277387a1 | ||
|
|
029a5f03fb | ||
|
|
2958843c5a | ||
|
|
daa95bdf02 | ||
|
|
316fffdd7c | ||
|
|
c65e2b2165 | ||
|
|
44b4842849 | ||
|
|
947c68f0a8 | ||
|
|
927c51a2bb | ||
|
|
07265e54ed | ||
|
|
00b00e6e77 | ||
|
|
1eabfebeb0 | ||
|
|
b9b6ee02c5 | ||
|
|
64f680bd70 | ||
|
|
e833835beb | ||
|
|
c358d28ca6 | ||
|
|
ba285f6ee8 | ||
|
|
72d7a158b2 | ||
|
|
c8f550a222 | ||
|
|
7324634620 | ||
|
|
b1d595be27 | ||
|
|
8e9def0e7e | ||
|
|
4141cefdec | ||
|
|
ffcdc70902 | ||
|
|
445173072d | ||
|
|
bb7b3c1a95 | ||
|
|
3d255abdaf | ||
|
|
308663a9fd | ||
|
|
b793060c05 | ||
|
|
008636f6fa | ||
|
|
467aaa1e07 | ||
|
|
8e79ba1efa | ||
|
|
d222085520 | ||
|
|
3d128c346e | ||
|
|
c4e387b408 | ||
|
|
788e477803 | ||
|
|
5983abbf68 | ||
|
|
2b096d0470 | ||
|
|
712bef33d8 | ||
|
|
7148869bac | ||
|
|
68ef61997d | ||
|
|
12ba1f793e | ||
|
|
e42b7e8c4c | ||
|
|
a8e491c22a | ||
|
|
570b5f4abf | ||
|
|
7bf0c76554 | ||
|
|
bf9ea8e515 | ||
|
|
39260d1b60 | ||
|
|
1c7ea0ab08 | ||
|
|
7416ec273a | ||
|
|
7bebf8ccfd | ||
|
|
29f039ff97 | ||
|
|
c45f41d171 | ||
|
|
a83e8c34c6 | ||
|
|
1e5e5e2e91 | ||
|
|
31025040dc | ||
|
|
ccc5fdb306 | ||
|
|
8bbf1388b1 | ||
|
|
33649d5d8d | ||
|
|
53bb36722c | ||
|
|
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 | ||
|
|
339a0db992 | ||
|
|
8efc1d139b | ||
|
|
94ea44f5fe | ||
|
|
3c12cfb17b | ||
|
|
ef9ad9a6fd | ||
|
|
4a98c430d4 | ||
|
|
432f4ea3bc | ||
|
|
9d875fd412 | ||
|
|
a2692da265 | ||
|
|
251acca5d5 | ||
|
|
e4c78bc9fa | ||
|
|
75fead124d | ||
|
|
155521ea11 | ||
|
|
81355cd289 | ||
|
|
88fd0640ad | ||
|
|
9904d7206e | ||
|
|
e552dd3506 | ||
|
|
568fd5d7aa | ||
|
|
9a2440a58a | ||
|
|
d4458e88ad | ||
|
|
079635b3ad | ||
|
|
035b9f7020 | ||
|
|
457daf5fc1 | ||
|
|
a64c2521de | ||
|
|
791a603f34 | ||
|
|
8729639907 | ||
|
|
ec9992c0ba | ||
|
|
4f5a370999 | ||
|
|
a856ef260b | ||
|
|
7613384396 | ||
|
|
fd484488ca | ||
|
|
b2a9eea9e2 | ||
|
|
58be70d34a | ||
|
|
97c639f902 | ||
|
|
a14c65566a | ||
|
|
6b66a6e980 | ||
|
|
a18d01bd4b | ||
|
|
f1b64604c0 | ||
|
|
7c3be096f1 | ||
|
|
d2ce081264 | ||
|
|
3c6895b295 | ||
|
|
2395a605b4 | ||
|
|
6c86c730c5 | ||
|
|
0815698aff | ||
|
|
4e0ede6fb8 | ||
|
|
4de6c17ae1 | ||
|
|
fa53509c36 | ||
|
|
2273d59166 | ||
|
|
2a82919aa1 | ||
|
|
60f6c7e585 | ||
|
|
31c09ae8dc | ||
|
|
07b2572541 | ||
|
|
7021c7ddea | ||
|
|
8c9f67f2f8 | ||
|
|
96a58ce851 | ||
|
|
5e0f15e4cc | ||
|
|
b8c0923f23 | ||
|
|
2e3cdcec99 | ||
|
|
0deff1fb44 | ||
|
|
34c76bb565 | ||
|
|
92b3faa23c | ||
|
|
430588dbdc | ||
|
|
786b98964c | ||
|
|
fad637580c | ||
|
|
e7f12e4b92 | ||
|
|
de375d2c89 | ||
|
|
fe79d77b62 | ||
|
|
7cb1d952b8 | ||
|
|
df07d9c308 | ||
|
|
97bf254d85 | ||
|
|
381fe85e39 | ||
|
|
a418f21ff1 | ||
|
|
f723e5f1f4 | ||
|
|
8080fc083e | ||
|
|
e1f6de8bd3 | ||
|
|
735d80d72f | ||
|
|
3eb97d81ec | ||
|
|
27e1428488 | ||
|
|
7b63198a08 | ||
|
|
45c947ddc1 | ||
|
|
e41458b7cd | ||
|
|
03ab12c64b | ||
|
|
f183c92661 | ||
|
|
109dacd121 | ||
|
|
0bff861d25 | ||
|
|
51be8954a9 | ||
|
|
4bd0201900 | ||
|
|
074b118bff | ||
|
|
1a4c9421dc | ||
|
|
6168a9a47a | ||
|
|
b6cd9b8138 | ||
|
|
cc153bfd06 | ||
|
|
cf3a36e108 | ||
|
|
2e6e26193f | ||
|
|
ae8ec346fd | ||
|
|
caf408a550 | ||
|
|
4543443a82 | ||
|
|
ae813adb18 | ||
|
|
6a27f35b2a | ||
|
|
1f9fbc2543 | ||
|
|
52f8dfc91c | ||
|
|
a5305e266f | ||
|
|
507ff901b4 | ||
|
|
8b5e2ce3a4 | ||
|
|
ebd2362c1c | ||
|
|
72f40744dd | ||
|
|
43f9e32282 | ||
|
|
f85a0f36ae | ||
|
|
1f2b333a8a | ||
|
|
4f056283b3 | ||
|
|
054d177ef2 | ||
|
|
84f441dbeb | ||
|
|
b3be06ea00 | ||
|
|
1fa7f545fd | ||
|
|
32f3e4cefa | ||
|
|
5109d98167 | ||
|
|
dc12f67ae4 | ||
|
|
8c34f22b7e | ||
|
|
ffa9ac6d5e | ||
|
|
0b9e63571e | ||
|
|
3b69be714e | ||
|
|
2ae9c0b2ae | ||
|
|
ef9fe2c506 | ||
|
|
9653304df2 | ||
|
|
b7dc19cb8d | ||
|
|
16bdac5240 | ||
|
|
53a7cd866f | ||
|
|
78fd7fb89f | ||
|
|
94ebb0bde7 | ||
|
|
681d34b775 | ||
|
|
d8ca7d3f04 | ||
|
|
bf68901255 | ||
|
|
ab949c2a0b | ||
|
|
90f7b1ad39 | ||
|
|
4b959cb797 | ||
|
|
670c4c845a | ||
|
|
531d05de64 | ||
|
|
3a929134ba | ||
|
|
f1aa143b96 | ||
|
|
f05e8f75cd | ||
|
|
190f2511af | ||
|
|
4ecc79977a | ||
|
|
7e4b6d4cac | ||
|
|
e023c7c42f | ||
|
|
a857ad5455 | ||
|
|
51460691bb | ||
|
|
01b0ef3eb8 | ||
|
|
4f006376c1 | ||
|
|
01db1e9805 | ||
|
|
a989129816 | ||
|
|
4ceee9a453 | ||
|
|
c90578311b | ||
|
|
7cd125ee33 | ||
|
|
76e97842a2 | ||
|
|
92aef0228e | ||
|
|
ca32144e80 | ||
|
|
e8ef69917a | ||
|
|
8a36085425 | ||
|
|
1916f5b2af | ||
|
|
c69957d1b8 | ||
|
|
426d1e77c6 | ||
|
|
ec681b5abc | ||
|
|
57ff03707c | ||
|
|
761fdd4d7a | ||
|
|
939e864a4a | ||
|
|
84c593016b | ||
|
|
e0a456724d | ||
|
|
c8a15d4d4d | ||
|
|
34e0bfc156 | ||
|
|
dfbc0446c9 | ||
|
|
0b9a101b08 | ||
|
|
1b9f2426a4 | ||
|
|
3ab8517320 | ||
|
|
77151d1752 | ||
|
|
b73731159f | ||
|
|
bfd004b73a | ||
|
|
71d2a32f2f | ||
|
|
1e4e88c05e | ||
|
|
1f4484f25f | ||
|
|
ab5143cb2f | ||
|
|
5fab70b931 | ||
|
|
3be25df928 | ||
|
|
74f0e28e8e | ||
|
|
f66aa05afb | ||
|
|
66d275d667 | ||
|
|
34211bfac9 | ||
|
|
1380621113 | ||
|
|
47baf5ea91 | ||
|
|
61ed2eaaa8 | ||
|
|
d55ee992f3 | ||
|
|
9a378b5763 | ||
|
|
c4ba025b17 | ||
|
|
4ca9b30b94 | ||
|
|
6e0af82475 | ||
|
|
7494ff7f89 | ||
|
|
327e849575 | ||
|
|
a0ad81a332 | ||
|
|
0847972b5a | ||
|
|
1e9c30733d | ||
|
|
90657e3a6b | ||
|
|
8ca0e6e380 | ||
|
|
7934eed0ab | ||
|
|
1809361301 | ||
|
|
01828e504d | ||
|
|
a2bb00faf1 |
52
.github/workflows/ci.yml
vendored
Normal file
52
.github/workflows/ci.yml
vendored
Normal file
@@ -0,0 +1,52 @@
|
||||
name: CI
|
||||
|
||||
on:
|
||||
push:
|
||||
paths-ignore:
|
||||
- '**/*.md'
|
||||
pull_request:
|
||||
paths-ignore:
|
||||
- '**/*.md'
|
||||
|
||||
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 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,4 +15,4 @@ script:
|
||||
language: java
|
||||
|
||||
jdk:
|
||||
- oraclejdk11
|
||||
- oraclejdk13
|
||||
17
README.md
17
README.md
@@ -1,21 +1,8 @@
|
||||
# Example Code Repository
|
||||
|
||||
[](https://travis-ci.org/thombergs/code-examples)
|
||||
[](https://github.com/thombergs/code-examples/actions?query=workflow%3ACI)
|
||||
|
||||
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/)
|
||||
@@ -1,5 +1,5 @@
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.0.1-bin.zip
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-bin.zip
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
|
||||
@@ -8,7 +8,7 @@ public class HelloWorldController {
|
||||
|
||||
@GetMapping("/hello")
|
||||
public String helloWorld(){
|
||||
return "Hello AWS!";
|
||||
return "Hello AWS (v4)!";
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
5
aws/aws-rds-hello-world/Dockerfile
Normal file
5
aws/aws-rds-hello-world/Dockerfile
Normal file
@@ -0,0 +1,5 @@
|
||||
FROM openjdk:8-jdk-alpine
|
||||
ARG JAR_FILE=build/libs/*.jar
|
||||
COPY ${JAR_FILE} app.jar
|
||||
ENTRYPOINT ["java","-jar","/app.jar"]
|
||||
EXPOSE 8080
|
||||
40
aws/aws-rds-hello-world/README.md
Normal file
40
aws/aws-rds-hello-world/README.md
Normal file
@@ -0,0 +1,40 @@
|
||||
# RDS Hello World Application
|
||||
|
||||
This is a simple Spring Boot application which requires access to a PostgreSQL database.
|
||||
|
||||
The application has a single endpoint `/hello` which prints out if the database connection was successful.
|
||||
|
||||
Get it in a Docker image via `docker pull reflectoring/aws-rds-hello-world`.
|
||||
|
||||
Use the image instead of your real application to test AWS CloudFormation stacks which need access to a database.
|
||||
|
||||
## Testing AWS RDS connectivity with this application
|
||||
|
||||
1. Create an RDS PostgreSQL database with the AWS console.
|
||||
2. Note the endpoint of your RDS database in the AWS console.
|
||||
3. Deploy the Docker container `reflectoring/aws-rds-hello-world` into AWS instead of your real application (this could be via a CloudFormation stack, manually, or however you are deploying your app).
|
||||
4. Configure your deployment in a way that Docker will pass the coordinates to your RDS database as environment variables, equivalent to this command:
|
||||
```
|
||||
docker run \
|
||||
-e SPRING_DATASOURCE_URL=':'<RDS-ENDPOINT>:5432/postgres \
|
||||
-e SPRING_DATASOURCE_USERNAME=<USERNAME> \
|
||||
-e SPRING_DATASOURCE_PASSWORD=<PASSWORD> \
|
||||
-p 8080:8080 reflectoring/aws-rds-hello-world
|
||||
```
|
||||
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/)
|
||||
|
||||
31
aws/aws-rds-hello-world/build.gradle
Normal file
31
aws/aws-rds-hello-world/build.gradle
Normal file
@@ -0,0 +1,31 @@
|
||||
plugins {
|
||||
id 'org.springframework.boot' version '2.2.4.RELEASE'
|
||||
id 'io.spring.dependency-management' version '1.0.9.RELEASE'
|
||||
id 'java'
|
||||
}
|
||||
|
||||
group = 'io.reflectoring'
|
||||
version = '0.0.1-SNAPSHOT'
|
||||
sourceCompatibility = '1.8'
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
implementation 'org.springframework.boot:spring-boot-starter-web'
|
||||
|
||||
// database
|
||||
implementation 'org.flywaydb:flyway-core'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-data-jdbc'
|
||||
runtimeOnly 'org.postgresql:postgresql'
|
||||
testRuntimeOnly 'org.postgresql:postgresql'
|
||||
|
||||
testImplementation('org.springframework.boot:spring-boot-starter-test') {
|
||||
exclude group: 'org.junit.vintage', module: 'junit-vintage-engine'
|
||||
}
|
||||
}
|
||||
|
||||
test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
18
aws/aws-rds-hello-world/docker-compose.yml
Normal file
18
aws/aws-rds-hello-world/docker-compose.yml
Normal file
@@ -0,0 +1,18 @@
|
||||
version: '3.7'
|
||||
|
||||
services:
|
||||
|
||||
postgres:
|
||||
container_name: "rds-hello-world"
|
||||
image: postgres
|
||||
volumes:
|
||||
- rds-hello-world:/var/lib/postgresql/data
|
||||
ports:
|
||||
- 5432:5432
|
||||
environment:
|
||||
- POSTGRES_USER=hello
|
||||
- POSTGRES_PASSWORD=hello
|
||||
|
||||
volumes:
|
||||
rds-hello-world:
|
||||
driver: local
|
||||
BIN
aws/aws-rds-hello-world/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
aws/aws-rds-hello-world/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
5
aws/aws-rds-hello-world/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
5
aws/aws-rds-hello-world/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-bin.zip
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
172
aws/aws-rds-hello-world/gradlew
vendored
Executable file
172
aws/aws-rds-hello-world/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
aws/aws-rds-hello-world/gradlew.bat
vendored
Executable file
84
aws/aws-rds-hello-world/gradlew.bat
vendored
Executable 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
aws/aws-rds-hello-world/settings.gradle
Normal file
1
aws/aws-rds-hello-world/settings.gradle
Normal file
@@ -0,0 +1 @@
|
||||
rootProject.name = 'aws-rds-hello-world'
|
||||
@@ -0,0 +1,13 @@
|
||||
package io.reflectoring.awshelloworld;
|
||||
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
|
||||
@SpringBootApplication
|
||||
public class AwsHelloWorldApplication {
|
||||
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(AwsHelloWorldApplication.class, args);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,23 @@
|
||||
package io.reflectoring.awshelloworld;
|
||||
|
||||
import org.springframework.web.bind.annotation.GetMapping;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
@RestController
|
||||
class HelloWorldController {
|
||||
|
||||
private final UserRepository userRepository;
|
||||
|
||||
HelloWorldController(UserRepository userRepository) {
|
||||
this.userRepository = userRepository;
|
||||
}
|
||||
|
||||
@GetMapping("/hello")
|
||||
String helloWorld(){
|
||||
|
||||
Iterable<User> users = userRepository.findAll();
|
||||
|
||||
return "Hello AWS! Successfully connected to the database!";
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,38 @@
|
||||
package io.reflectoring.awshelloworld;
|
||||
|
||||
import org.springframework.data.annotation.Id;
|
||||
import org.springframework.data.relational.core.mapping.Table;
|
||||
|
||||
@Table("hello_user")
|
||||
public class User {
|
||||
|
||||
@Id
|
||||
private Long id;
|
||||
|
||||
private String name;
|
||||
|
||||
public User(Long id, String name) {
|
||||
this.id = id;
|
||||
this.name = name;
|
||||
}
|
||||
|
||||
public User() {
|
||||
|
||||
}
|
||||
|
||||
public Long getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(Long id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(String name) {
|
||||
this.name = name;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
package io.reflectoring.awshelloworld;
|
||||
|
||||
import org.springframework.data.repository.CrudRepository;
|
||||
|
||||
public interface UserRepository extends CrudRepository<User, Long> {
|
||||
}
|
||||
@@ -0,0 +1,5 @@
|
||||
spring:
|
||||
datasource:
|
||||
url: jdbc:postgresql://localhost:5432/hello
|
||||
username: hello
|
||||
password: hello
|
||||
@@ -0,0 +1,5 @@
|
||||
create table hello_user (
|
||||
id varchar(36) not null unique,
|
||||
name varchar(100) not null,
|
||||
primary key(id)
|
||||
);
|
||||
117
aws/cdk/.mvn/wrapper/MavenWrapperDownloader.java
vendored
Normal file
117
aws/cdk/.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
aws/cdk/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
BIN
aws/cdk/.mvn/wrapper/maven-wrapper.jar
vendored
Normal file
Binary file not shown.
2
aws/cdk/.mvn/wrapper/maven-wrapper.properties
vendored
Normal file
2
aws/cdk/.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
|
||||
0
aws/cdk/mvnw
vendored
Normal file
0
aws/cdk/mvnw
vendored
Normal file
182
aws/cdk/mvnw.cmd
vendored
Normal file
182
aws/cdk/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%
|
||||
23
aws/cdk/src/main/java/com/myorg/SecondStack.java
Normal file
23
aws/cdk/src/main/java/com/myorg/SecondStack.java
Normal file
@@ -0,0 +1,23 @@
|
||||
package com.myorg;
|
||||
|
||||
import software.amazon.awscdk.core.Construct;
|
||||
import software.amazon.awscdk.core.Stack;
|
||||
import software.amazon.awscdk.core.StackProps;
|
||||
import software.amazon.awscdk.services.s3.Bucket;
|
||||
|
||||
public class SecondStack extends Stack {
|
||||
public SecondStack(final Construct scope, final String id) {
|
||||
this(scope, id, null);
|
||||
}
|
||||
|
||||
public SecondStack(final Construct scope, final String id, final StackProps props) {
|
||||
super(scope, id, props);
|
||||
|
||||
String bucketName = (String) this.getNode().tryGetContext("bucketName2");
|
||||
|
||||
Bucket bucket = Bucket.Builder.create(this, "bucket")
|
||||
.bucketName(bucketName)
|
||||
.build();
|
||||
|
||||
}
|
||||
}
|
||||
13
aws/cloudformation/ecs-in-two-public-subnets/README.md
Normal file
13
aws/cloudformation/ecs-in-two-public-subnets/README.md
Normal file
@@ -0,0 +1,13 @@
|
||||
# Overview
|
||||
|
||||

|
||||
|
||||
## 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/)
|
||||
|
||||
@@ -0,0 +1 @@
|
||||
<mxfile modified="2020-04-30T21:28:18.347Z" host="app.diagrams.net" agent="5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36" etag="X9Eef2gSXNvxkCnmY-_0" version="13.0.4" type="device"><diagram id="Ht1M8jgEwFfnCIfOTk4-" name="Page-1">7Vpbb+o4EP41PC7K/fLIrWcr9UiVWO05+4RM4garIc4aU6C/fseJc7ND6ekB9nQXVKmZsTMez3yf7TEM7Ml6/4WhfPWVxjgdWEa8H9jTgWWZphXCP6E5lBrPcEpFwkgsOzWKOXnFUmlI7ZbEeNPpyClNOcm7yohmGY54R4cYo7tutyeadkfNUYI1xTxCqa79RmK+qublhU3D75gkKzl0YPllwxpVneVMNisU011LZc8G9oRRysun9X6CUxG8Ki7le3dHWmvHGM74e15Y4cjav97736avy2fP+Rr/vVj+Vjn3gtKtnPGfjxPpMD9UUcgpyXgRSXcMf/DSxBi40DIR0tByFYUq+12FqUvCRlehyn5XYarmTWV8U3WwpdCkjnlDGd9oOQh/9phueUoyPKkxZ4AyYSgmkIsJTSkDXUYziN54xdcpSCY87laE43mOIhHVHfAFdE804xL1plXJMvDiHUBNLp7X+0QQbIh2G2eYMLrNiyHvAfe9rYuXPBKvc0afceXSwLItJwhMRwxE0lRx9QUzTgD6o5QkwiqnYhAkpRQ/cWER/CdZ8lBIU9uQPreGGI3G/jgAfYw2KxzL8EicwRB4fxTBZs0LWFAwXWPODtClesGTaJVrSQXeXUNMO5S6VYuTjiGVSC4GSW264Qs8SMr8AH3MUKPP43aZkkjQZ7vMML9R6fNTaYOjLSP8sGg6zwteSXffRzLQz8I7e+adj2n1OOdmmm0pTLPCoR1obKsZ2Gab6ZkFSC5COPvGtxvf/vt8c4Buftj6uL8A90xfI99sMgfFHLMXAshQqddzxNBS5468SeC1Q2oezZcKMCU7talzJMTvJsTvXf9MuycHVnip04arxX9geamA7hIeEvEwynNYDREnwK2qjVWNDxTFuhZogLIIs6oFXKvNaSntXT46bKhS94CWOH2kG1L4Yk+XlHO6PsnFCBIEvnQWn56FxB6iZqaLFCa2WFbz0JaBOzdwbef4OncGvMAZs4MX0w+GgQ6Yqmps48W3LgSX4DRc7kWwi13TQ2uRhmy5yYuIqCj5gjjeocMnAwmR81sk0v1+bHgXxUa9SrSw4esL+lWxodct5VL+B9o8vy+f6tZoW3difC2OIsZ+MDPa/JsSBobKlGeUiQioeZka7gQ2nJ4d46n4XAFRp843aJOX4Xgie+FH/4GH4Q3dsgiXx50xiH0HHxxtzgO2utatwGb07lx+z8blX6xKdm5o+3+gzXL/fbRZeo2oYQzHCa6CKxBBE5qhdNZo25nBWTwSN8kivymNnoUqXRZyleYCOYjxqp/cPODNO5LWO5R29nXcsSeQCmVOFtfggtizw/e28JcQoKqT4nTfbpwe2tIjZgTiKNDVKk7EfN9OMYSnAM5bgZWbBkw0wfzUwUPHTPvgXOGG4RSOcC9d5/ogIc09imq+wV+owM9Uap9yTvKl9t24YsdU6i9bvTEs56wZKgBaT/EnMOvdMPsRzJ6Eons1KFrqUvhhLAYKFs0rY1Ev8z8ZFveEf6/GhucWEkFqgCiEQ0u44NLpvhOu5UnpKng1TfXKyfgYXq1Tho7gFXIt6sm6m7yoPeqw7fYT7JhfWv/QUehSenBe8vQV3TfyXIU84dW4Y1ff+9VfjzjDIDDqj2LwvUxyHLdr1hOn6aNmfxFemc41eKVfWLx1mfUp76isH7+dqn9T0nwtUv1KxT7TxaaS7MDvu7vyw6FUtks81zlOrCMlHojNL2JK8DS/K7Jn/wA=</diagram></mxfile>
|
||||
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 25 KiB |
247
aws/cloudformation/ecs-in-two-public-subnets/network.yml
Normal file
247
aws/cloudformation/ecs-in-two-public-subnets/network.yml
Normal file
@@ -0,0 +1,247 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: A network stack for deploying containers in AWS ECS.
|
||||
This stack creates a VPC with two public subnets and a loadbalancer to balance traffic between those subnets.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Resources:
|
||||
|
||||
VPC:
|
||||
Type: AWS::EC2::VPC
|
||||
Properties:
|
||||
CidrBlock: '10.0.0.0/16'
|
||||
|
||||
PublicSubnetOne:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 0
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.1.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
PublicSubnetTwo:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 1
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.2.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
InternetGateway:
|
||||
Type: AWS::EC2::InternetGateway
|
||||
|
||||
GatewayAttachement:
|
||||
Type: AWS::EC2::VPCGatewayAttachment
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
InternetGatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicRouteTable:
|
||||
Type: AWS::EC2::RouteTable
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicSubnetOneRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetOne
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicSubnetTwoRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetTwo
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicRoute:
|
||||
Type: AWS::EC2::Route
|
||||
DependsOn: GatewayAttachement
|
||||
Properties:
|
||||
RouteTableId: !Ref 'PublicRouteTable'
|
||||
DestinationCidrBlock: '0.0.0.0/0'
|
||||
GatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicLoadBalancerSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the public facing load balancer
|
||||
VpcId: !Ref 'VPC'
|
||||
SecurityGroupIngress:
|
||||
# Allow access to ALB from anywhere on the internet
|
||||
- CidrIp: 0.0.0.0/0
|
||||
IpProtocol: -1
|
||||
|
||||
PublicLoadBalancer:
|
||||
Type: AWS::ElasticLoadBalancingV2::LoadBalancer
|
||||
Properties:
|
||||
Scheme: internet-facing
|
||||
Subnets:
|
||||
# The load balancer is placed into the public subnets, so that traffic
|
||||
# from the internet can reach the load balancer directly via the internet gateway
|
||||
- !Ref PublicSubnetOne
|
||||
- !Ref PublicSubnetTwo
|
||||
SecurityGroups: [!Ref 'PublicLoadBalancerSecurityGroup']
|
||||
|
||||
DummyTargetGroupPublic:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: 6
|
||||
HealthCheckPath: /
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
Name: "no-op"
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicLoadBalancerListener:
|
||||
Type: AWS::ElasticLoadBalancingV2::Listener
|
||||
DependsOn:
|
||||
- PublicLoadBalancer
|
||||
Properties:
|
||||
DefaultActions:
|
||||
- TargetGroupArn: !Ref 'DummyTargetGroupPublic'
|
||||
Type: 'forward'
|
||||
LoadBalancerArn: !Ref 'PublicLoadBalancer'
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
|
||||
ECSCluster:
|
||||
Type: AWS::ECS::Cluster
|
||||
|
||||
ECSSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the ECS containers
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
ECSSecurityGroupIngressFromPublicALB:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from the public ALB
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'PublicLoadBalancerSecurityGroup'
|
||||
|
||||
ECSSecurityGroupIngressFromSelf:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from other containers in the same security group
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'ECSSecurityGroup'
|
||||
|
||||
ECSRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: ecs-service
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Rules which allow ECS to attach network interfaces to instances
|
||||
# on your behalf in order for awsvpc networking mode to work right
|
||||
- 'ec2:AttachNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterfacePermission'
|
||||
- 'ec2:DeleteNetworkInterface'
|
||||
- 'ec2:DeleteNetworkInterfacePermission'
|
||||
- 'ec2:Describe*'
|
||||
- 'ec2:DetachNetworkInterface'
|
||||
|
||||
# Rules which allow ECS to update load balancers on your behalf
|
||||
# with the information sabout how to send traffic to your containers
|
||||
- 'elasticloadbalancing:DeregisterInstancesFromLoadBalancer'
|
||||
- 'elasticloadbalancing:DeregisterTargets'
|
||||
- 'elasticloadbalancing:Describe*'
|
||||
- 'elasticloadbalancing:RegisterInstancesWithLoadBalancer'
|
||||
- 'elasticloadbalancing:RegisterTargets'
|
||||
Resource: '*'
|
||||
|
||||
ECSTaskExecutionRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs-tasks.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: AmazonECSTaskExecutionRolePolicy
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Allow the ECS Tasks to download images from ECR
|
||||
- 'ecr:GetAuthorizationToken'
|
||||
- 'ecr:BatchCheckLayerAvailability'
|
||||
- 'ecr:GetDownloadUrlForLayer'
|
||||
- 'ecr:BatchGetImage'
|
||||
|
||||
# Allow the ECS tasks to upload logs to CloudWatch
|
||||
- 'logs:CreateLogStream'
|
||||
- 'logs:PutLogEvents'
|
||||
Resource: '*'
|
||||
|
||||
Outputs:
|
||||
ClusterName:
|
||||
Description: The name of the ECS cluster
|
||||
Value: !Ref 'ECSCluster'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ClusterName' ] ]
|
||||
ExternalUrl:
|
||||
Description: The url of the external load balancer
|
||||
Value: !Join ['', ['http://', !GetAtt 'PublicLoadBalancer.DNSName']]
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ExternalUrl' ] ]
|
||||
ECSRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSRole' ] ]
|
||||
ECSTaskExecutionRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSTaskExecutionRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSTaskExecutionRole' ] ]
|
||||
PublicListener:
|
||||
Description: The ARN of the public load balancer's Listener
|
||||
Value: !Ref PublicLoadBalancerListener
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicListener' ] ]
|
||||
VPCId:
|
||||
Description: The ID of the VPC that this stack is deployed in
|
||||
Value: !Ref 'VPC'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'VPCId' ] ]
|
||||
PublicSubnetOne:
|
||||
Description: Public subnet one
|
||||
Value: !Ref 'PublicSubnetOne'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetOne' ] ]
|
||||
PublicSubnetTwo:
|
||||
Description: Public subnet two
|
||||
Value: !Ref 'PublicSubnetTwo'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetTwo' ] ]
|
||||
ECSSecurityGroup:
|
||||
Description: A security group used to allow ECS containers to receive traffic
|
||||
Value: !Ref 'ECSSecurityGroup'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSSecurityGroup' ] ]
|
||||
139
aws/cloudformation/ecs-in-two-public-subnets/service.yml
Normal file
139
aws/cloudformation/ecs-in-two-public-subnets/service.yml
Normal file
@@ -0,0 +1,139 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: Deploy a service on AWS Fargate, hosted in two public subnets and accessible via a public load balancer.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Parameters:
|
||||
StackName:
|
||||
Type: String
|
||||
Description: The name of the networking stack that
|
||||
these resources are put into.
|
||||
ServiceName:
|
||||
Type: String
|
||||
Description: A human-readable name for the service.
|
||||
HealthCheckPath:
|
||||
Type: String
|
||||
Default: /health
|
||||
Description: Path to perform the healthcheck on each instance.
|
||||
HealthCheckIntervalSeconds:
|
||||
Type: Number
|
||||
Default: 5
|
||||
Description: Number of seconds to wait between each health check.
|
||||
ImageUrl:
|
||||
Type: String
|
||||
Description: The url of a docker image that will handle incoming traffic.
|
||||
ContainerPort:
|
||||
Type: Number
|
||||
Default: 80
|
||||
Description: The port number the application inside the docker container
|
||||
is binding to.
|
||||
ContainerCpu:
|
||||
Type: Number
|
||||
Default: 256
|
||||
Description: How much CPU to give the container. 1024 is 1 CPU.
|
||||
ContainerMemory:
|
||||
Type: Number
|
||||
Default: 512
|
||||
Description: How much memory in megabytes to give the container.
|
||||
Path:
|
||||
Type: String
|
||||
Default: "*"
|
||||
Description: A path on the public load balancer that this service
|
||||
should be connected to.
|
||||
DesiredCount:
|
||||
Type: Number
|
||||
Default: 2
|
||||
Description: How many copies of the service task to run.
|
||||
|
||||
Resources:
|
||||
|
||||
TargetGroup:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: !Ref 'HealthCheckIntervalSeconds'
|
||||
HealthCheckPath: !Ref 'HealthCheckPath'
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
TargetType: ip
|
||||
Name: !Ref 'ServiceName'
|
||||
Port: !Ref 'ContainerPort'
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'VPCId']]
|
||||
|
||||
LoadBalancerRule:
|
||||
Type: AWS::ElasticLoadBalancingV2::ListenerRule
|
||||
Properties:
|
||||
Actions:
|
||||
- TargetGroupArn: !Ref 'TargetGroup'
|
||||
Type: 'forward'
|
||||
Conditions:
|
||||
- Field: path-pattern
|
||||
Values: [!Ref 'Path']
|
||||
ListenerArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicListener']]
|
||||
Priority: 1
|
||||
|
||||
LogGroup:
|
||||
Type: AWS::Logs::LogGroup
|
||||
Properties:
|
||||
LogGroupName: !Ref 'ServiceName'
|
||||
RetentionInDays: 1
|
||||
|
||||
TaskDefinition:
|
||||
Type: AWS::ECS::TaskDefinition
|
||||
Properties:
|
||||
Family: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
NetworkMode: awsvpc
|
||||
RequiresCompatibilities:
|
||||
- FARGATE
|
||||
ExecutionRoleArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ECSTaskExecutionRole']]
|
||||
ContainerDefinitions:
|
||||
- Name: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
Image: !Ref 'ImageUrl'
|
||||
PortMappings:
|
||||
- ContainerPort: !Ref 'ContainerPort'
|
||||
LogConfiguration:
|
||||
LogDriver: 'awslogs'
|
||||
Options:
|
||||
awslogs-group: !Ref 'ServiceName'
|
||||
awslogs-region: !Ref AWS::Region
|
||||
awslogs-stream-prefix: !Ref 'ServiceName'
|
||||
|
||||
Service:
|
||||
Type: AWS::ECS::Service
|
||||
DependsOn: LoadBalancerRule
|
||||
Properties:
|
||||
ServiceName: !Ref 'ServiceName'
|
||||
Cluster:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ClusterName']]
|
||||
LaunchType: FARGATE
|
||||
DeploymentConfiguration:
|
||||
MaximumPercent: 200
|
||||
MinimumHealthyPercent: 50
|
||||
DesiredCount: !Ref 'DesiredCount'
|
||||
NetworkConfiguration:
|
||||
AwsvpcConfiguration:
|
||||
AssignPublicIp: ENABLED
|
||||
SecurityGroups:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ECSSecurityGroup']]
|
||||
Subnets:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicSubnetOne']]
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicSubnetTwo']]
|
||||
TaskDefinition: !Ref 'TaskDefinition'
|
||||
LoadBalancers:
|
||||
- ContainerName: !Ref 'ServiceName'
|
||||
ContainerPort: !Ref 'ContainerPort'
|
||||
TargetGroupArn: !Ref 'TargetGroup'
|
||||
15
aws/cloudformation/ecs-zero-downtime-deployment/README.md
Normal file
15
aws/cloudformation/ecs-zero-downtime-deployment/README.md
Normal file
@@ -0,0 +1,15 @@
|
||||
# Overview
|
||||
|
||||

|
||||
|
||||
## 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/)
|
||||
|
||||
|
||||
|
||||
18
aws/cloudformation/ecs-zero-downtime-deployment/create-change-set.sh
Executable file
18
aws/cloudformation/ecs-zero-downtime-deployment/create-change-set.sh
Executable file
@@ -0,0 +1,18 @@
|
||||
# Turning off the AWS pager so that the CLI doesn't open an editor for each command result
|
||||
export AWS_PAGER=""
|
||||
|
||||
aws cloudformation create-change-set \
|
||||
--change-set-name update-reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--use-previous-template \
|
||||
--parameters \
|
||||
ParameterKey=StackName,ParameterValue=reflectoring-ecs-zero-downtime-deployment-network \
|
||||
ParameterKey=ServiceName,ParameterValue=reflectoring-hello-world \
|
||||
ParameterKey=ImageUrl,ParameterValue=docker.io/reflectoring/aws-hello-world:v4 \
|
||||
ParameterKey=ContainerPort,ParameterValue=8080 \
|
||||
ParameterKey=HealthCheckPath,ParameterValue=/hello \
|
||||
ParameterKey=HealthCheckIntervalSeconds,ParameterValue=90
|
||||
|
||||
aws cloudformation describe-change-set \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--change-set-name update-reflectoring-ecs-zero-downtime-deployment-service
|
||||
22
aws/cloudformation/ecs-zero-downtime-deployment/create.sh
Executable file
22
aws/cloudformation/ecs-zero-downtime-deployment/create.sh
Executable file
@@ -0,0 +1,22 @@
|
||||
# Turning off the AWS pager so that the CLI doesn't open an editor for each command result
|
||||
export AWS_PAGER=""
|
||||
|
||||
aws cloudformation create-stack \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-network \
|
||||
--template-body file://network.yml \
|
||||
--capabilities CAPABILITY_IAM
|
||||
|
||||
aws cloudformation wait stack-create-complete --stack-name reflectoring-ecs-zero-downtime-deployment-network
|
||||
|
||||
aws cloudformation create-stack \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--template-body file://service.yml \
|
||||
--parameters \
|
||||
ParameterKey=StackName,ParameterValue=reflectoring-ecs-zero-downtime-deployment-network \
|
||||
ParameterKey=ServiceName,ParameterValue=reflectoring-hello-world \
|
||||
ParameterKey=ImageUrl,ParameterValue=docker.io/reflectoring/aws-hello-world:v3 \
|
||||
ParameterKey=ContainerPort,ParameterValue=8080 \
|
||||
ParameterKey=HealthCheckPath,ParameterValue=/hello \
|
||||
ParameterKey=HealthCheckIntervalSeconds,ParameterValue=90
|
||||
|
||||
aws cloudformation wait stack-create-complete --stack-name reflectoring-ecs-zero-downtime-deployment-service
|
||||
8
aws/cloudformation/ecs-zero-downtime-deployment/delete.sh
Executable file
8
aws/cloudformation/ecs-zero-downtime-deployment/delete.sh
Executable file
@@ -0,0 +1,8 @@
|
||||
# Turning off the AWS pager so that the CLI doesn't open an editor for each command result
|
||||
export AWS_PAGER=""
|
||||
|
||||
aws cloudformation delete-stack --stack-name reflectoring-ecs-zero-downtime-deployment-service
|
||||
aws cloudformation wait stack-delete-complete --stack-name reflectoring-ecs-zero-downtime-deployment-service
|
||||
|
||||
aws cloudformation delete-stack --stack-name reflectoring-ecs-zero-downtime-deployment-network
|
||||
aws cloudformation wait stack-delete-complete --stack-name reflectoring-ecs-zero-downtime-deployment-network
|
||||
@@ -0,0 +1 @@
|
||||
<mxfile modified="2020-04-30T21:28:18.347Z" host="app.diagrams.net" agent="5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36" etag="X9Eef2gSXNvxkCnmY-_0" version="13.0.4" type="device"><diagram id="Ht1M8jgEwFfnCIfOTk4-" name="Page-1">7Vpbb+o4EP41PC7K/fLIrWcr9UiVWO05+4RM4garIc4aU6C/fseJc7ND6ekB9nQXVKmZsTMez3yf7TEM7Ml6/4WhfPWVxjgdWEa8H9jTgWWZphXCP6E5lBrPcEpFwkgsOzWKOXnFUmlI7ZbEeNPpyClNOcm7yohmGY54R4cYo7tutyeadkfNUYI1xTxCqa79RmK+qublhU3D75gkKzl0YPllwxpVneVMNisU011LZc8G9oRRysun9X6CUxG8Ki7le3dHWmvHGM74e15Y4cjav97736avy2fP+Rr/vVj+Vjn3gtKtnPGfjxPpMD9UUcgpyXgRSXcMf/DSxBi40DIR0tByFYUq+12FqUvCRlehyn5XYarmTWV8U3WwpdCkjnlDGd9oOQh/9phueUoyPKkxZ4AyYSgmkIsJTSkDXUYziN54xdcpSCY87laE43mOIhHVHfAFdE804xL1plXJMvDiHUBNLp7X+0QQbIh2G2eYMLrNiyHvAfe9rYuXPBKvc0afceXSwLItJwhMRwxE0lRx9QUzTgD6o5QkwiqnYhAkpRQ/cWER/CdZ8lBIU9uQPreGGI3G/jgAfYw2KxzL8EicwRB4fxTBZs0LWFAwXWPODtClesGTaJVrSQXeXUNMO5S6VYuTjiGVSC4GSW264Qs8SMr8AH3MUKPP43aZkkjQZ7vMML9R6fNTaYOjLSP8sGg6zwteSXffRzLQz8I7e+adj2n1OOdmmm0pTLPCoR1obKsZ2Gab6ZkFSC5COPvGtxvf/vt8c4Buftj6uL8A90xfI99sMgfFHLMXAshQqddzxNBS5468SeC1Q2oezZcKMCU7talzJMTvJsTvXf9MuycHVnip04arxX9geamA7hIeEvEwynNYDREnwK2qjVWNDxTFuhZogLIIs6oFXKvNaSntXT46bKhS94CWOH2kG1L4Yk+XlHO6PsnFCBIEvnQWn56FxB6iZqaLFCa2WFbz0JaBOzdwbef4OncGvMAZs4MX0w+GgQ6Yqmps48W3LgSX4DRc7kWwi13TQ2uRhmy5yYuIqCj5gjjeocMnAwmR81sk0v1+bHgXxUa9SrSw4esL+lWxodct5VL+B9o8vy+f6tZoW3difC2OIsZ+MDPa/JsSBobKlGeUiQioeZka7gQ2nJ4d46n4XAFRp843aJOX4Xgie+FH/4GH4Q3dsgiXx50xiH0HHxxtzgO2utatwGb07lx+z8blX6xKdm5o+3+gzXL/fbRZeo2oYQzHCa6CKxBBE5qhdNZo25nBWTwSN8kivymNnoUqXRZyleYCOYjxqp/cPODNO5LWO5R29nXcsSeQCmVOFtfggtizw/e28JcQoKqT4nTfbpwe2tIjZgTiKNDVKk7EfN9OMYSnAM5bgZWbBkw0wfzUwUPHTPvgXOGG4RSOcC9d5/ogIc09imq+wV+owM9Uap9yTvKl9t24YsdU6i9bvTEs56wZKgBaT/EnMOvdMPsRzJ6Eons1KFrqUvhhLAYKFs0rY1Ev8z8ZFveEf6/GhucWEkFqgCiEQ0u44NLpvhOu5UnpKng1TfXKyfgYXq1Tho7gFXIt6sm6m7yoPeqw7fYT7JhfWv/QUehSenBe8vQV3TfyXIU84dW4Y1ff+9VfjzjDIDDqj2LwvUxyHLdr1hOn6aNmfxFemc41eKVfWLx1mfUp76isH7+dqn9T0nwtUv1KxT7TxaaS7MDvu7vyw6FUtks81zlOrCMlHojNL2JK8DS/K7Jn/wA=</diagram></mxfile>
|
||||
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 25 KiB |
8
aws/cloudformation/ecs-zero-downtime-deployment/execute-change-set.sh
Executable file
8
aws/cloudformation/ecs-zero-downtime-deployment/execute-change-set.sh
Executable file
@@ -0,0 +1,8 @@
|
||||
# Turning off the AWS pager so that the CLI doesn't open an editor for each command result
|
||||
export AWS_PAGER=""
|
||||
|
||||
aws cloudformation execute-change-set \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--change-set-name update-reflectoring-ecs-zero-downtime-deployment-service
|
||||
|
||||
aws cloudformation wait stack-update-complete --stack-name reflectoring-ecs-zero-downtime-deployment-service
|
||||
247
aws/cloudformation/ecs-zero-downtime-deployment/network.yml
Normal file
247
aws/cloudformation/ecs-zero-downtime-deployment/network.yml
Normal file
@@ -0,0 +1,247 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: A network stack for deploying containers in AWS ECS.
|
||||
This stack creates a VPC with two public subnets and a loadbalancer to balance traffic between those subnets.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Resources:
|
||||
|
||||
VPC:
|
||||
Type: AWS::EC2::VPC
|
||||
Properties:
|
||||
CidrBlock: '10.0.0.0/16'
|
||||
|
||||
PublicSubnetOne:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 0
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.1.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
PublicSubnetTwo:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 1
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.2.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
InternetGateway:
|
||||
Type: AWS::EC2::InternetGateway
|
||||
|
||||
GatewayAttachement:
|
||||
Type: AWS::EC2::VPCGatewayAttachment
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
InternetGatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicRouteTable:
|
||||
Type: AWS::EC2::RouteTable
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicSubnetOneRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetOne
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicSubnetTwoRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetTwo
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicRoute:
|
||||
Type: AWS::EC2::Route
|
||||
DependsOn: GatewayAttachement
|
||||
Properties:
|
||||
RouteTableId: !Ref 'PublicRouteTable'
|
||||
DestinationCidrBlock: '0.0.0.0/0'
|
||||
GatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicLoadBalancerSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the public facing load balancer
|
||||
VpcId: !Ref 'VPC'
|
||||
SecurityGroupIngress:
|
||||
# Allow access to ALB from anywhere on the internet
|
||||
- CidrIp: 0.0.0.0/0
|
||||
IpProtocol: -1
|
||||
|
||||
PublicLoadBalancer:
|
||||
Type: AWS::ElasticLoadBalancingV2::LoadBalancer
|
||||
Properties:
|
||||
Scheme: internet-facing
|
||||
Subnets:
|
||||
# The load balancer is placed into the public subnets, so that traffic
|
||||
# from the internet can reach the load balancer directly via the internet gateway
|
||||
- !Ref PublicSubnetOne
|
||||
- !Ref PublicSubnetTwo
|
||||
SecurityGroups: [!Ref 'PublicLoadBalancerSecurityGroup']
|
||||
|
||||
DummyTargetGroupPublic:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: 6
|
||||
HealthCheckPath: /
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
Name: "no-op"
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicLoadBalancerListener:
|
||||
Type: AWS::ElasticLoadBalancingV2::Listener
|
||||
DependsOn:
|
||||
- PublicLoadBalancer
|
||||
Properties:
|
||||
DefaultActions:
|
||||
- TargetGroupArn: !Ref 'DummyTargetGroupPublic'
|
||||
Type: 'forward'
|
||||
LoadBalancerArn: !Ref 'PublicLoadBalancer'
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
|
||||
ECSCluster:
|
||||
Type: AWS::ECS::Cluster
|
||||
|
||||
ECSSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the ECS containers
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
ECSSecurityGroupIngressFromPublicALB:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from the public ALB
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'PublicLoadBalancerSecurityGroup'
|
||||
|
||||
ECSSecurityGroupIngressFromSelf:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from other containers in the same security group
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'ECSSecurityGroup'
|
||||
|
||||
ECSRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: ecs-service
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Rules which allow ECS to attach network interfaces to instances
|
||||
# on your behalf in order for awsvpc networking mode to work right
|
||||
- 'ec2:AttachNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterfacePermission'
|
||||
- 'ec2:DeleteNetworkInterface'
|
||||
- 'ec2:DeleteNetworkInterfacePermission'
|
||||
- 'ec2:Describe*'
|
||||
- 'ec2:DetachNetworkInterface'
|
||||
|
||||
# Rules which allow ECS to update load balancers on your behalf
|
||||
# with the information sabout how to send traffic to your containers
|
||||
- 'elasticloadbalancing:DeregisterInstancesFromLoadBalancer'
|
||||
- 'elasticloadbalancing:DeregisterTargets'
|
||||
- 'elasticloadbalancing:Describe*'
|
||||
- 'elasticloadbalancing:RegisterInstancesWithLoadBalancer'
|
||||
- 'elasticloadbalancing:RegisterTargets'
|
||||
Resource: '*'
|
||||
|
||||
ECSTaskExecutionRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs-tasks.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: AmazonECSTaskExecutionRolePolicy
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Allow the ECS Tasks to download images from ECR
|
||||
- 'ecr:GetAuthorizationToken'
|
||||
- 'ecr:BatchCheckLayerAvailability'
|
||||
- 'ecr:GetDownloadUrlForLayer'
|
||||
- 'ecr:BatchGetImage'
|
||||
|
||||
# Allow the ECS tasks to upload logs to CloudWatch
|
||||
- 'logs:CreateLogStream'
|
||||
- 'logs:PutLogEvents'
|
||||
Resource: '*'
|
||||
|
||||
Outputs:
|
||||
ClusterName:
|
||||
Description: The name of the ECS cluster
|
||||
Value: !Ref 'ECSCluster'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ClusterName' ] ]
|
||||
ExternalUrl:
|
||||
Description: The url of the external load balancer
|
||||
Value: !Join ['', ['http://', !GetAtt 'PublicLoadBalancer.DNSName']]
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ExternalUrl' ] ]
|
||||
ECSRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSRole' ] ]
|
||||
ECSTaskExecutionRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSTaskExecutionRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSTaskExecutionRole' ] ]
|
||||
PublicListener:
|
||||
Description: The ARN of the public load balancer's Listener
|
||||
Value: !Ref PublicLoadBalancerListener
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicListener' ] ]
|
||||
VPCId:
|
||||
Description: The ID of the VPC that this stack is deployed in
|
||||
Value: !Ref 'VPC'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'VPCId' ] ]
|
||||
PublicSubnetOne:
|
||||
Description: Public subnet one
|
||||
Value: !Ref 'PublicSubnetOne'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetOne' ] ]
|
||||
PublicSubnetTwo:
|
||||
Description: Public subnet two
|
||||
Value: !Ref 'PublicSubnetTwo'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetTwo' ] ]
|
||||
ECSSecurityGroup:
|
||||
Description: A security group used to allow ECS containers to receive traffic
|
||||
Value: !Ref 'ECSSecurityGroup'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSSecurityGroup' ] ]
|
||||
139
aws/cloudformation/ecs-zero-downtime-deployment/service.yml
Normal file
139
aws/cloudformation/ecs-zero-downtime-deployment/service.yml
Normal file
@@ -0,0 +1,139 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: Deploy a service on AWS Fargate, hosted in two public subnets and accessible via a public load balancer.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Parameters:
|
||||
StackName:
|
||||
Type: String
|
||||
Description: The name of the networking stack that
|
||||
these resources are put into.
|
||||
ServiceName:
|
||||
Type: String
|
||||
Description: A human-readable name for the service.
|
||||
HealthCheckPath:
|
||||
Type: String
|
||||
Default: /health
|
||||
Description: Path to perform the healthcheck on each instance.
|
||||
HealthCheckIntervalSeconds:
|
||||
Type: Number
|
||||
Default: 5
|
||||
Description: Number of seconds to wait between each health check.
|
||||
ImageUrl:
|
||||
Type: String
|
||||
Description: The url of a docker image that will handle incoming traffic.
|
||||
ContainerPort:
|
||||
Type: Number
|
||||
Default: 80
|
||||
Description: The port number the application inside the docker container
|
||||
is binding to.
|
||||
ContainerCpu:
|
||||
Type: Number
|
||||
Default: 256
|
||||
Description: How much CPU to give the container. 1024 is 1 CPU.
|
||||
ContainerMemory:
|
||||
Type: Number
|
||||
Default: 512
|
||||
Description: How much memory in megabytes to give the container.
|
||||
Path:
|
||||
Type: String
|
||||
Default: "*"
|
||||
Description: A path on the public load balancer that this service
|
||||
should be connected to.
|
||||
DesiredCount:
|
||||
Type: Number
|
||||
Default: 2
|
||||
Description: How many copies of the service task to run.
|
||||
|
||||
Resources:
|
||||
|
||||
TargetGroup:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: !Ref 'HealthCheckIntervalSeconds'
|
||||
HealthCheckPath: !Ref 'HealthCheckPath'
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
TargetType: ip
|
||||
Name: !Ref 'ServiceName'
|
||||
Port: !Ref 'ContainerPort'
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'VPCId']]
|
||||
|
||||
LoadBalancerRule:
|
||||
Type: AWS::ElasticLoadBalancingV2::ListenerRule
|
||||
Properties:
|
||||
Actions:
|
||||
- TargetGroupArn: !Ref 'TargetGroup'
|
||||
Type: 'forward'
|
||||
Conditions:
|
||||
- Field: path-pattern
|
||||
Values: [!Ref 'Path']
|
||||
ListenerArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicListener']]
|
||||
Priority: 1
|
||||
|
||||
LogGroup:
|
||||
Type: AWS::Logs::LogGroup
|
||||
Properties:
|
||||
LogGroupName: !Ref 'ServiceName'
|
||||
RetentionInDays: 1
|
||||
|
||||
TaskDefinition:
|
||||
Type: AWS::ECS::TaskDefinition
|
||||
Properties:
|
||||
Family: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
NetworkMode: awsvpc
|
||||
RequiresCompatibilities:
|
||||
- FARGATE
|
||||
ExecutionRoleArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ECSTaskExecutionRole']]
|
||||
ContainerDefinitions:
|
||||
- Name: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
Image: !Ref 'ImageUrl'
|
||||
PortMappings:
|
||||
- ContainerPort: !Ref 'ContainerPort'
|
||||
LogConfiguration:
|
||||
LogDriver: 'awslogs'
|
||||
Options:
|
||||
awslogs-group: !Ref 'ServiceName'
|
||||
awslogs-region: !Ref AWS::Region
|
||||
awslogs-stream-prefix: !Ref 'ServiceName'
|
||||
|
||||
Service:
|
||||
Type: AWS::ECS::Service
|
||||
DependsOn: LoadBalancerRule
|
||||
Properties:
|
||||
ServiceName: !Ref 'ServiceName'
|
||||
Cluster:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ClusterName']]
|
||||
LaunchType: FARGATE
|
||||
DeploymentConfiguration:
|
||||
MaximumPercent: 200
|
||||
MinimumHealthyPercent: 50
|
||||
DesiredCount: !Ref 'DesiredCount'
|
||||
NetworkConfiguration:
|
||||
AwsvpcConfiguration:
|
||||
AssignPublicIp: ENABLED
|
||||
SecurityGroups:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ECSSecurityGroup']]
|
||||
Subnets:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicSubnetOne']]
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicSubnetTwo']]
|
||||
TaskDefinition: !Ref 'TaskDefinition'
|
||||
LoadBalancers:
|
||||
- ContainerName: !Ref 'ServiceName'
|
||||
ContainerPort: !Ref 'ContainerPort'
|
||||
TargetGroupArn: !Ref 'TargetGroup'
|
||||
109
aws/cloudformation/ecs-zero-downtime-deployment/task.yml
Normal file
109
aws/cloudformation/ecs-zero-downtime-deployment/task.yml
Normal file
@@ -0,0 +1,109 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: Deploy a service on AWS Fargate, hosted in two public subnets and accessible via a public load balancer.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Parameters:
|
||||
StackName:
|
||||
Type: String
|
||||
Description: The name of the networking stack that
|
||||
these resources are put into.
|
||||
ServiceName:
|
||||
Type: String
|
||||
Description: A human-readable name for the service.
|
||||
HealthCheckPath:
|
||||
Type: String
|
||||
Default: /health
|
||||
Description: Path to perform the healthcheck on each instance.
|
||||
HealthCheckIntervalSeconds:
|
||||
Type: Number
|
||||
Default: 5
|
||||
Description: Number of seconds to wait between each health check.
|
||||
ImageUrl:
|
||||
Type: String
|
||||
Description: The url of a docker image that will handle incoming traffic.
|
||||
ContainerPort:
|
||||
Type: Number
|
||||
Default: 80
|
||||
Description: The port number the application inside the docker container
|
||||
is binding to.
|
||||
ContainerCpu:
|
||||
Type: Number
|
||||
Default: 256
|
||||
Description: How much CPU to give the container. 1024 is 1 CPU.
|
||||
ContainerMemory:
|
||||
Type: Number
|
||||
Default: 512
|
||||
Description: How much memory in megabytes to give the container.
|
||||
Path:
|
||||
Type: String
|
||||
Default: "*"
|
||||
Description: A path on the public load balancer that this service
|
||||
should be connected to.
|
||||
DesiredCount:
|
||||
Type: Number
|
||||
Default: 2
|
||||
Description: How many copies of the service task to run.
|
||||
|
||||
Resources:
|
||||
|
||||
TargetGroup:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: !Ref 'HealthCheckIntervalSeconds'
|
||||
HealthCheckPath: !Ref 'HealthCheckPath'
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
TargetType: ip
|
||||
Name: !Ref 'ServiceName'
|
||||
Port: !Ref 'ContainerPort'
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'VPCId']]
|
||||
|
||||
LoadBalancerRule:
|
||||
Type: AWS::ElasticLoadBalancingV2::ListenerRule
|
||||
Properties:
|
||||
Actions:
|
||||
- TargetGroupArn: !Ref 'TargetGroup'
|
||||
Type: 'forward'
|
||||
Conditions:
|
||||
- Field: path-pattern
|
||||
Values: [!Ref 'Path']
|
||||
ListenerArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'PublicListener']]
|
||||
Priority: 1
|
||||
|
||||
LogGroup:
|
||||
Type: AWS::Logs::LogGroup
|
||||
Properties:
|
||||
LogGroupName: !Ref 'ServiceName'
|
||||
RetentionInDays: 1
|
||||
|
||||
TaskDefinition:
|
||||
Type: AWS::ECS::TaskDefinition
|
||||
Properties:
|
||||
Family: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
NetworkMode: awsvpc
|
||||
RequiresCompatibilities:
|
||||
- FARGATE
|
||||
ExecutionRoleArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'StackName', 'ECSTaskExecutionRole']]
|
||||
ContainerDefinitions:
|
||||
- Name: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
Image: !Ref 'ImageUrl'
|
||||
PortMappings:
|
||||
- ContainerPort: !Ref 'ContainerPort'
|
||||
LogConfiguration:
|
||||
LogDriver: 'awslogs'
|
||||
Options:
|
||||
awslogs-group: !Ref 'ServiceName'
|
||||
awslogs-region: !Ref AWS::Region
|
||||
awslogs-stream-prefix: !Ref 'ServiceName'
|
||||
17
aws/cloudformation/ecs-zero-downtime-deployment/update.sh
Executable file
17
aws/cloudformation/ecs-zero-downtime-deployment/update.sh
Executable file
@@ -0,0 +1,17 @@
|
||||
# Turning off the AWS pager so that the CLI doesn't open an editor for each command result
|
||||
export AWS_PAGER=""
|
||||
|
||||
IMAGE_URL=$1
|
||||
|
||||
aws cloudformation update-stack \
|
||||
--stack-name reflectoring-ecs-zero-downtime-deployment-service \
|
||||
--use-previous-template \
|
||||
--parameters \
|
||||
ParameterKey=StackName,ParameterValue=reflectoring-ecs-zero-downtime-deployment-network \
|
||||
ParameterKey=ServiceName,ParameterValue=reflectoring-hello-world \
|
||||
ParameterKey=ImageUrl,ParameterValue=$IMAGE_URL \
|
||||
ParameterKey=ContainerPort,ParameterValue=8080 \
|
||||
ParameterKey=HealthCheckPath,ParameterValue=/hello \
|
||||
ParameterKey=HealthCheckIntervalSeconds,ParameterValue=90
|
||||
|
||||
aws cloudformation wait stack-update-complete --stack-name reflectoring-ecs-zero-downtime-deployment-service
|
||||
7
aws/cloudformation/rds-in-private-subnet/README.md
Normal file
7
aws/cloudformation/rds-in-private-subnet/README.md
Normal file
@@ -0,0 +1,7 @@
|
||||
# Overview
|
||||
|
||||

|
||||
|
||||
# Companion Blog Post
|
||||
|
||||
[The AWS Journey Part 3: Connecting a Spring Boot Application to an RDS Instance with CloudFormation](https://reflectoring.io/aws-cloudformation-rds/)
|
||||
88
aws/cloudformation/rds-in-private-subnet/database.yml
Normal file
88
aws/cloudformation/rds-in-private-subnet/database.yml
Normal file
@@ -0,0 +1,88 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: A stack that creates an RDS instance and places it into two subnets
|
||||
Parameters:
|
||||
NetworkStackName:
|
||||
Type: String
|
||||
Description: The name of the networking stack that this stack will build upon.
|
||||
DBInstanceClass:
|
||||
Type: String
|
||||
Description: The ID of the second subnet to place the RDS instance into.
|
||||
Default: 'db.t2.micro'
|
||||
DBName:
|
||||
Type: String
|
||||
Description: The name of the database that is created within the PostgreSQL instance.
|
||||
DBUsername:
|
||||
Type: String
|
||||
Description: The master user name for the PostgreSQL instance.
|
||||
Resources:
|
||||
|
||||
Secret:
|
||||
Type: "AWS::SecretsManager::Secret"
|
||||
Properties:
|
||||
Name: !Ref 'DBUsername'
|
||||
GenerateSecretString:
|
||||
# This will generate a JSON object with the keys "username" and password.
|
||||
SecretStringTemplate: !Join ['', ['{"username": "', !Ref 'DBUsername' ,'"}']]
|
||||
GenerateStringKey: "password"
|
||||
PasswordLength: 32
|
||||
ExcludeCharacters: '"@/\'
|
||||
|
||||
DBSubnetGroup:
|
||||
Type: AWS::RDS::DBSubnetGroup
|
||||
Properties:
|
||||
DBSubnetGroupDescription: Subnet group for the RDS instance
|
||||
DBSubnetGroupName: DBSubnetGroup
|
||||
SubnetIds:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'PrivateSubnetOne']]
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'PrivateSubnetTwo']]
|
||||
|
||||
PostgresInstance:
|
||||
Type: AWS::RDS::DBInstance
|
||||
Properties:
|
||||
AllocatedStorage: 20
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 0
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
DBInstanceClass: !Ref 'DBInstanceClass'
|
||||
DBName: !Ref 'DBName'
|
||||
DBSubnetGroupName: !Ref 'DBSubnetGroup'
|
||||
Engine: postgres
|
||||
EngineVersion: 11.5
|
||||
MasterUsername: !Ref 'DBUsername'
|
||||
MasterUserPassword: !Join ['', ['{{resolve:secretsmanager:', !Ref Secret, ':SecretString:password}}' ]]
|
||||
PubliclyAccessible: false
|
||||
VPCSecurityGroups:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'DBSecurityGroupId']]
|
||||
|
||||
SecretRDSInstanceAttachment:
|
||||
Type: "AWS::SecretsManager::SecretTargetAttachment"
|
||||
Properties:
|
||||
SecretId: !Ref Secret
|
||||
TargetId: !Ref PostgresInstance
|
||||
TargetType: AWS::RDS::DBInstance
|
||||
|
||||
Outputs:
|
||||
EndpointAddress:
|
||||
Description: Address of the RDS endpoint.
|
||||
Value: !GetAtt 'PostgresInstance.Endpoint.Address'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'EndpointAddress' ] ]
|
||||
EndpointPort:
|
||||
Description: Port of the RDS endpoint.
|
||||
Value: !GetAtt 'PostgresInstance.Endpoint.Port'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'EndpointPort' ] ]
|
||||
DBName:
|
||||
Description: The name of the database that is created within the PostgreSQL instance.
|
||||
Value: !Ref DBName
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'DBName' ] ]
|
||||
Secret:
|
||||
Description: Reference to the secret containing the password to the database.
|
||||
Value: !Ref 'Secret'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'Secret' ] ]
|
||||
298
aws/cloudformation/rds-in-private-subnet/network.yml
Normal file
298
aws/cloudformation/rds-in-private-subnet/network.yml
Normal file
@@ -0,0 +1,298 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: A network stack for deploying containers in AWS ECS.
|
||||
This stack creates a VPC with two public subnets and a loadbalancer to balance traffic between those subnets.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Resources:
|
||||
|
||||
VPC:
|
||||
Type: AWS::EC2::VPC
|
||||
Properties:
|
||||
CidrBlock: '10.0.0.0/16'
|
||||
|
||||
PublicSubnetOne:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 0
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.1.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
PublicSubnetTwo:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 1
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.2.0/24'
|
||||
MapPublicIpOnLaunch: true
|
||||
|
||||
InternetGateway:
|
||||
Type: AWS::EC2::InternetGateway
|
||||
|
||||
GatewayAttachement:
|
||||
Type: AWS::EC2::VPCGatewayAttachment
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
InternetGatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicRouteTable:
|
||||
Type: AWS::EC2::RouteTable
|
||||
Properties:
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicSubnetOneRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetOne
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicSubnetTwoRouteTableAssociation:
|
||||
Type: AWS::EC2::SubnetRouteTableAssociation
|
||||
Properties:
|
||||
SubnetId: !Ref PublicSubnetTwo
|
||||
RouteTableId: !Ref PublicRouteTable
|
||||
|
||||
PublicRoute:
|
||||
Type: AWS::EC2::Route
|
||||
DependsOn: GatewayAttachement
|
||||
Properties:
|
||||
RouteTableId: !Ref 'PublicRouteTable'
|
||||
DestinationCidrBlock: '0.0.0.0/0'
|
||||
GatewayId: !Ref 'InternetGateway'
|
||||
|
||||
PublicLoadBalancerSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the public facing load balancer
|
||||
VpcId: !Ref 'VPC'
|
||||
SecurityGroupIngress:
|
||||
# Allow access to ALB from anywhere on the internet
|
||||
- CidrIp: 0.0.0.0/0
|
||||
IpProtocol: -1
|
||||
|
||||
PublicLoadBalancer:
|
||||
Type: AWS::ElasticLoadBalancingV2::LoadBalancer
|
||||
Properties:
|
||||
Scheme: internet-facing
|
||||
Subnets:
|
||||
# The load balancer is placed into the public subnets, so that traffic
|
||||
# from the internet can reach the load balancer directly via the internet gateway
|
||||
- !Ref PublicSubnetOne
|
||||
- !Ref PublicSubnetTwo
|
||||
SecurityGroups: [!Ref 'PublicLoadBalancerSecurityGroup']
|
||||
|
||||
DummyTargetGroupPublic:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: 6
|
||||
HealthCheckPath: /
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
Name: "no-op"
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
PublicLoadBalancerListener:
|
||||
Type: AWS::ElasticLoadBalancingV2::Listener
|
||||
DependsOn:
|
||||
- PublicLoadBalancer
|
||||
Properties:
|
||||
DefaultActions:
|
||||
- TargetGroupArn: !Ref 'DummyTargetGroupPublic'
|
||||
Type: 'forward'
|
||||
LoadBalancerArn: !Ref 'PublicLoadBalancer'
|
||||
Port: 80
|
||||
Protocol: HTTP
|
||||
|
||||
ECSCluster:
|
||||
Type: AWS::ECS::Cluster
|
||||
|
||||
ECSSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the ECS containers
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
ECSSecurityGroupIngressFromPublicALB:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from the public ALB
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'PublicLoadBalancerSecurityGroup'
|
||||
|
||||
ECSSecurityGroupIngressFromSelf:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from other containers in the same security group
|
||||
GroupId: !Ref 'ECSSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'ECSSecurityGroup'
|
||||
|
||||
ECSRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: ecs-service
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Rules which allow ECS to attach network interfaces to instances
|
||||
# on your behalf in order for awsvpc networking mode to work right
|
||||
- 'ec2:AttachNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterface'
|
||||
- 'ec2:CreateNetworkInterfacePermission'
|
||||
- 'ec2:DeleteNetworkInterface'
|
||||
- 'ec2:DeleteNetworkInterfacePermission'
|
||||
- 'ec2:Describe*'
|
||||
- 'ec2:DetachNetworkInterface'
|
||||
|
||||
# Rules which allow ECS to update load balancers on your behalf
|
||||
# with the information sabout how to send traffic to your containers
|
||||
- 'elasticloadbalancing:DeregisterInstancesFromLoadBalancer'
|
||||
- 'elasticloadbalancing:DeregisterTargets'
|
||||
- 'elasticloadbalancing:Describe*'
|
||||
- 'elasticloadbalancing:RegisterInstancesWithLoadBalancer'
|
||||
- 'elasticloadbalancing:RegisterTargets'
|
||||
Resource: '*'
|
||||
|
||||
ECSTaskExecutionRole:
|
||||
Type: AWS::IAM::Role
|
||||
Properties:
|
||||
AssumeRolePolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Principal:
|
||||
Service: [ecs-tasks.amazonaws.com]
|
||||
Action: ['sts:AssumeRole']
|
||||
Path: /
|
||||
Policies:
|
||||
- PolicyName: AmazonECSTaskExecutionRolePolicy
|
||||
PolicyDocument:
|
||||
Statement:
|
||||
- Effect: Allow
|
||||
Action:
|
||||
# Allow the ECS Tasks to download images from ECR
|
||||
- 'ecr:GetAuthorizationToken'
|
||||
- 'ecr:BatchCheckLayerAvailability'
|
||||
- 'ecr:GetDownloadUrlForLayer'
|
||||
- 'ecr:BatchGetImage'
|
||||
|
||||
# Allow the ECS tasks to upload logs to CloudWatch
|
||||
- 'logs:CreateLogStream'
|
||||
- 'logs:PutLogEvents'
|
||||
Resource: '*'
|
||||
|
||||
PrivateSubnetOne:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 0
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.101.0/24'
|
||||
MapPublicIpOnLaunch: false
|
||||
|
||||
PrivateSubnetTwo:
|
||||
Type: AWS::EC2::Subnet
|
||||
Properties:
|
||||
AvailabilityZone:
|
||||
Fn::Select:
|
||||
- 1
|
||||
- Fn::GetAZs: {Ref: 'AWS::Region'}
|
||||
VpcId: !Ref 'VPC'
|
||||
CidrBlock: '10.0.102.0/24'
|
||||
MapPublicIpOnLaunch: false
|
||||
|
||||
DBSecurityGroup:
|
||||
Type: AWS::EC2::SecurityGroup
|
||||
Properties:
|
||||
GroupDescription: Access to the RDS instance
|
||||
VpcId: !Ref 'VPC'
|
||||
|
||||
DBSecurityGroupIngressFromECS:
|
||||
Type: AWS::EC2::SecurityGroupIngress
|
||||
Properties:
|
||||
Description: Ingress from the ECS containers to the RDS instance
|
||||
GroupId: !Ref 'DBSecurityGroup'
|
||||
IpProtocol: -1
|
||||
SourceSecurityGroupId: !Ref 'ECSSecurityGroup'
|
||||
|
||||
Outputs:
|
||||
PrivateSubnetOne:
|
||||
Description: Private subnet one
|
||||
Value: !Ref 'PrivateSubnetOne'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PrivateSubnetOne' ] ]
|
||||
PrivateSubnetTwo:
|
||||
Description: Private subnet two
|
||||
Value: !Ref 'PrivateSubnetTwo'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PrivateSubnetTwo' ] ]
|
||||
DBSecurityGroupId:
|
||||
Description: ID of the security group that an RDS instance can be placed into.
|
||||
Value: !Ref 'DBSecurityGroup'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'DBSecurityGroupId' ] ]
|
||||
ClusterName:
|
||||
Description: The name of the ECS cluster
|
||||
Value: !Ref 'ECSCluster'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ClusterName' ] ]
|
||||
ExternalUrl:
|
||||
Description: The url of the external load balancer
|
||||
Value: !Join ['', ['http://', !GetAtt 'PublicLoadBalancer.DNSName']]
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ExternalUrl' ] ]
|
||||
ECSRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSRole' ] ]
|
||||
ECSTaskExecutionRole:
|
||||
Description: The ARN of the ECS role
|
||||
Value: !GetAtt 'ECSTaskExecutionRole.Arn'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSTaskExecutionRole' ] ]
|
||||
PublicListener:
|
||||
Description: The ARN of the public load balancer's Listener
|
||||
Value: !Ref PublicLoadBalancerListener
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicListener' ] ]
|
||||
VPCId:
|
||||
Description: The ID of the VPC that this stack is deployed in
|
||||
Value: !Ref 'VPC'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'VPCId' ] ]
|
||||
PublicSubnetOne:
|
||||
Description: Public subnet one
|
||||
Value: !Ref 'PublicSubnetOne'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetOne' ] ]
|
||||
PublicSubnetTwo:
|
||||
Description: Public subnet two
|
||||
Value: !Ref 'PublicSubnetTwo'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'PublicSubnetTwo' ] ]
|
||||
ECSSecurityGroup:
|
||||
Description: A security group used to allow ECS containers to receive traffic
|
||||
Value: !Ref 'ECSSecurityGroup'
|
||||
Export:
|
||||
Name: !Join [ ':', [ !Ref 'AWS::StackName', 'ECSSecurityGroup' ] ]
|
||||
@@ -0,0 +1 @@
|
||||
<mxfile modified="2020-05-11T21:18:00.281Z" host="app.diagrams.net" agent="5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36" etag="d22WT4Cb11cg7E6o-Mdr" version="13.0.9" type="device"><diagram id="Ht1M8jgEwFfnCIfOTk4-" name="Page-1">7VvZcuI4FP0aHpuyJK+PYUk6VemaTNHTyxNlsGI8MRYtiy1fP5KRwZbEkkBI00OKqljXspZ7z7mLDA3UHi/uaDgZfSERThvQihYN1GlACJDj8H9CslxJPM9eCWKaRLLTRtBLXrAUWlI6TSKc1zoyQlKWTOrCIckyPGQ1WUgpmde7PZG0PuskjLEm6A3DVJd+TyI2klLgBpsbn3ESj+TUPvRWN8Zh2VnuJB+FEZlXRKjbQG1KCFtdjRdtnArllXpZPXe75e56YRRn7JAHHuJp6+UXX9i3PB/cv7TDr5/+/SRHmYXpVG74cTpIk6FY73SQYSbXzpalQiYkyVihVKfFP3zOttVw+J22aDWhowjUtlcXAL0lxqgL1LZXFwB1eKDMD9QFVgRaqza8pcxvVRbIP6hFpixNMtxew8/iwpiGUcLN0iYpoVyWkYxrrzVi45S3AL+cjxKGe5NwKLQ659ThsieSMUkAAMu2VLx4hgNoIq7Hi1hwrRnOc7sZUzKdFFPecwoY7/ZzPJzShC37m849RskzlsvNi0a52AZE0PZ9YIslJGlakXeDW9R1uXyGKUs4SW7SJBaTMiKGDWUrxU9MDMu3l2TxQ9HqIEtuyTRPFOYjHMnlSEjyKfBiK9bBmkHc9WAyxowueRf5AHIk6aTXQaU7mVc47EvZqEJf4IICIYX7kK4jXg+/YRe/kAQzk22Eh3Dxcu9977wMnl37S/SrP/hUeoIK2749tq8Mu3yGzSbDg3kkl3oiCt3ctLyWb6aQxhcDq7ZSCLh1CgFLp5BrGyiEAut4/hiDla0HK5rMQoav0eoP4tL/L1o5wQdHq0U6ve3jv2af7eXPX2zwzwu+m5pSwyvbrmz7Hdh2VGBT2QbgmdlmjG2OxrYO36zVk0Sz7grzqIQz5BWawZwbt+27VUWCrVZSYaXYZD2UYs0T+EAXKlZxDFYptV9LN8AJ0g1juo6utfHV//15/k+tjc/u/4xkA57Gtm67JzwgprOEQ+G3dn3HGcQ/wPUhU6X1bq5Pj0YN6KZivwN+EYuLm8mE+8KQJZxZ5T1a3nwgYaRLue7CbIhpeYcvbT2cZl+j86hxobTjQzjA6SPJk2ItqDMgjJHxXiYOucX4Wmqux+BGUDPc7LSf8o31B+U+NCdw6/gOsrd7uROgBdp1tEDXagKkl+aBjhcPvhNc/P1wuRfKLmKmG46FGbJBPik0oqLkjpcY83B5YSBJ5P76sVy+GRvuu2Jj7SU22PhYYARbnPrXMH8+zJhqVETwVsyvKVEo2PO7VpV8nYTygVb2zggVGlCN0rGcNg89htjxVPydAU77Upswn6zU8ZQsxDrMuQ7FOZnSIV5lOi3eNOU8eJifBmk2UGKWryPNM4Qs770iFtSzdQ1gOIpxqVkBBxKTLEy7G2nVLDiLbsT7Q2HclAyfhSgdFO3SxgVsQsrKfpLI/MnbJF17Cy0psZ2WK2DKE84sWiOL24Iuf1QbP0WD59ey2VlUb3aW1dYjpgnXo4DW7jRxBZJdepSU5fuKMdvv84VOdwKmmsSUR8gUpzyczuqvWE2QkMM9irpqAz6lgIeBq+Wlq33K56ovRZWhgFJ2Il/PcVeq0MYqYLre6BHIda/IPQC5ewHpfBwgoXVCRCrVAArOj0hTPndJiHw7iIKPAxGyAhVEb0OQXX7npkSQbTWtyh84CEzcECIJX3eTZ1vblw81DuxcpdYfOAqWVys4LbL1hHRXpXKRBQh8femx/mrS5sSr/LITOlHVqhYmgZ4uBk3D8a5jH58wGl9vwf1I4NZiMcW9vx94x/uM+6dMHERdFCxolOt4gF0PFV8aeL9S1LXtmsWBZ6/DWNXoUDe5v8N1HmXySy8Rtpplb1ZvSOp3keJDwh9Qwh+0m77/tgjoAnNsOXXMc3bHsNf2PzrmGd8s6qi/nsH8LmcwR70yVc9gkGs3TXH1nY5hjGC7+Fp2kbAf5dz8ulLJ8tamkBWNZaVhLmMVP/o23+3ovns7zz/EVSPrrZWK6g8PLHTPXZvY1hn8tP4C9MKoc/pjoNfyZ1f8Oyz3OQt/7KBekEOVBE0L2bYPPcd3oeW9MQdybGVYHh38wHUc4PHQgdR3PqdKidz6bhTq7O2/L4VS64ojUyje3PwCZ9V98zsm1P0P</diagram></mxfile>
|
||||
File diff suppressed because one or more lines are too long
|
After Width: | Height: | Size: 32 KiB |
164
aws/cloudformation/rds-in-private-subnet/service.yml
Normal file
164
aws/cloudformation/rds-in-private-subnet/service.yml
Normal file
@@ -0,0 +1,164 @@
|
||||
AWSTemplateFormatVersion: '2010-09-09'
|
||||
Description: Deploy a service on AWS Fargate, hosted in two public subnets and accessible via a public load balancer.
|
||||
Derived from a template at https://github.com/nathanpeck/aws-cloudformation-fargate.
|
||||
Parameters:
|
||||
NetworkStackName:
|
||||
Type: String
|
||||
Description: The name of the networking stack that
|
||||
these resources are put into.
|
||||
DatabaseStackName:
|
||||
Type: String
|
||||
Description: The name of the database stack with the database this service should connect to.
|
||||
ServiceName:
|
||||
Type: String
|
||||
Description: A human-readable name for the service.
|
||||
HealthCheckPath:
|
||||
Type: String
|
||||
Default: /health
|
||||
Description: Path to perform the healthcheck on each instance.
|
||||
HealthCheckIntervalSeconds:
|
||||
Type: Number
|
||||
Default: 5
|
||||
Description: Number of seconds to wait between each health check.
|
||||
ImageUrl:
|
||||
Type: String
|
||||
Description: The url of a docker image that will handle incoming traffic.
|
||||
ContainerPort:
|
||||
Type: Number
|
||||
Default: 80
|
||||
Description: The port number the application inside the docker container
|
||||
is binding to.
|
||||
ContainerCpu:
|
||||
Type: Number
|
||||
Default: 256
|
||||
Description: How much CPU to give the container. 1024 is 1 CPU.
|
||||
ContainerMemory:
|
||||
Type: Number
|
||||
Default: 512
|
||||
Description: How much memory in megabytes to give the container.
|
||||
Path:
|
||||
Type: String
|
||||
Default: "*"
|
||||
Description: A path on the public load balancer that this service
|
||||
should be connected to.
|
||||
DesiredCount:
|
||||
Type: Number
|
||||
Default: 2
|
||||
Description: How many copies of the service task to run.
|
||||
|
||||
Resources:
|
||||
|
||||
TargetGroup:
|
||||
Type: AWS::ElasticLoadBalancingV2::TargetGroup
|
||||
Properties:
|
||||
HealthCheckIntervalSeconds: !Ref 'HealthCheckIntervalSeconds'
|
||||
HealthCheckPath: !Ref 'HealthCheckPath'
|
||||
HealthCheckProtocol: HTTP
|
||||
HealthCheckTimeoutSeconds: 5
|
||||
HealthyThresholdCount: 2
|
||||
TargetType: ip
|
||||
Name: !Ref 'ServiceName'
|
||||
Port: !Ref 'ContainerPort'
|
||||
Protocol: HTTP
|
||||
UnhealthyThresholdCount: 2
|
||||
VpcId:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'VPCId']]
|
||||
|
||||
LoadBalancerRule:
|
||||
Type: AWS::ElasticLoadBalancingV2::ListenerRule
|
||||
Properties:
|
||||
Actions:
|
||||
- TargetGroupArn: !Ref 'TargetGroup'
|
||||
Type: 'forward'
|
||||
Conditions:
|
||||
- Field: path-pattern
|
||||
Values: [!Ref 'Path']
|
||||
ListenerArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'PublicListener']]
|
||||
Priority: 1
|
||||
|
||||
LogGroup:
|
||||
Type: AWS::Logs::LogGroup
|
||||
Properties:
|
||||
LogGroupName: !Ref 'ServiceName'
|
||||
RetentionInDays: 1
|
||||
|
||||
TaskDefinition:
|
||||
Type: AWS::ECS::TaskDefinition
|
||||
Properties:
|
||||
Family: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
NetworkMode: awsvpc
|
||||
RequiresCompatibilities:
|
||||
- FARGATE
|
||||
ExecutionRoleArn:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'ECSTaskExecutionRole']]
|
||||
ContainerDefinitions:
|
||||
- Name: !Ref 'ServiceName'
|
||||
Cpu: !Ref 'ContainerCpu'
|
||||
Memory: !Ref 'ContainerMemory'
|
||||
Image: !Ref 'ImageUrl'
|
||||
Environment:
|
||||
- Name: SPRING_DATASOURCE_URL
|
||||
Value: !Join
|
||||
- ''
|
||||
- - 'jdbc:postgresql://'
|
||||
- Fn::ImportValue: !Join [':', [!Ref 'DatabaseStackName', 'EndpointAddress']]
|
||||
- ':'
|
||||
- Fn::ImportValue: !Join [':', [!Ref 'DatabaseStackName', 'EndpointPort']]
|
||||
- '/'
|
||||
- Fn::ImportValue: !Join [':', [!Ref 'DatabaseStackName', 'DBName']]
|
||||
- Name: SPRING_DATASOURCE_USERNAME
|
||||
Value: !Join
|
||||
- ''
|
||||
- - '{{resolve:secretsmanager:'
|
||||
- Fn::ImportValue: !Join [':', [!Ref 'DatabaseStackName', 'Secret']]
|
||||
- ':SecretString:username}}'
|
||||
- Name: SPRING_DATASOURCE_PASSWORD
|
||||
Value: !Join
|
||||
- ''
|
||||
- - '{{resolve:secretsmanager:'
|
||||
- Fn::ImportValue: !Join [':', [!Ref 'DatabaseStackName', 'Secret']]
|
||||
- ':SecretString:password}}'
|
||||
PortMappings:
|
||||
- ContainerPort: !Ref 'ContainerPort'
|
||||
LogConfiguration:
|
||||
LogDriver: 'awslogs'
|
||||
Options:
|
||||
awslogs-group: !Ref 'ServiceName'
|
||||
awslogs-region: !Ref AWS::Region
|
||||
awslogs-stream-prefix: !Ref 'ServiceName'
|
||||
|
||||
Service:
|
||||
Type: AWS::ECS::Service
|
||||
DependsOn: LoadBalancerRule
|
||||
Properties:
|
||||
ServiceName: !Ref 'ServiceName'
|
||||
Cluster:
|
||||
Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'ClusterName']]
|
||||
LaunchType: FARGATE
|
||||
DeploymentConfiguration:
|
||||
MaximumPercent: 200
|
||||
MinimumHealthyPercent: 50
|
||||
DesiredCount: !Ref 'DesiredCount'
|
||||
NetworkConfiguration:
|
||||
AwsvpcConfiguration:
|
||||
AssignPublicIp: ENABLED
|
||||
SecurityGroups:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'ECSSecurityGroup']]
|
||||
Subnets:
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'PublicSubnetOne']]
|
||||
- Fn::ImportValue:
|
||||
!Join [':', [!Ref 'NetworkStackName', 'PublicSubnetTwo']]
|
||||
TaskDefinition: !Ref 'TaskDefinition'
|
||||
LoadBalancers:
|
||||
- ContainerName: !Ref 'ServiceName'
|
||||
ContainerPort: !Ref 'ContainerPort'
|
||||
TargetGroupArn: !Ref 'TargetGroup'
|
||||
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%
|
||||
84
aws/localstack/pom.xml
Normal file
84
aws/localstack/pom.xml
Normal file
@@ -0,0 +1,84 @@
|
||||
<?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>
|
||||
<dependency>
|
||||
<groupId>org.junit.jupiter</groupId>
|
||||
<artifactId>junit-jupiter-engine</artifactId>
|
||||
<scope>test</scope>
|
||||
</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,103 @@
|
||||
/**
|
||||
*
|
||||
*/
|
||||
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.Disabled;
|
||||
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" })
|
||||
@Disabled(value = "Disabled because starting Docker containers like this does not run on GitHub Actions.")
|
||||
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,105 @@
|
||||
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.Disabled;
|
||||
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"})
|
||||
@Disabled(value = "Disabled because starting Docker containers like this does not run on GitHub Actions.")
|
||||
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();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
141
aws/s3/.gitignore
vendored
Normal file
141
aws/s3/.gitignore
vendored
Normal file
@@ -0,0 +1,141 @@
|
||||
|
||||
# Created by https://www.toptal.com/developers/gitignore/api/java,intellij+all,gradle
|
||||
# Edit at https://www.toptal.com/developers/gitignore?templates=java,intellij+all,gradle
|
||||
|
||||
### Intellij+all ###
|
||||
# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider
|
||||
# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
|
||||
|
||||
# User-specific stuff
|
||||
.idea/**/workspace.xml
|
||||
.idea/**/tasks.xml
|
||||
.idea/**/usage.statistics.xml
|
||||
.idea/**/dictionaries
|
||||
.idea/**/shelf
|
||||
|
||||
# Generated files
|
||||
.idea/**/contentModel.xml
|
||||
|
||||
# Sensitive or high-churn files
|
||||
.idea/**/dataSources/
|
||||
.idea/**/dataSources.ids
|
||||
.idea/**/dataSources.local.xml
|
||||
.idea/**/sqlDataSources.xml
|
||||
.idea/**/dynamic.xml
|
||||
.idea/**/uiDesigner.xml
|
||||
.idea/**/dbnavigator.xml
|
||||
|
||||
# Gradle
|
||||
.idea/**/gradle.xml
|
||||
.idea/**/libraries
|
||||
|
||||
# Gradle and Maven with auto-import
|
||||
# When using Gradle or Maven with auto-import, you should exclude module files,
|
||||
# since they will be recreated, and may cause churn. Uncomment if using
|
||||
# auto-import.
|
||||
# .idea/artifacts
|
||||
# .idea/compiler.xml
|
||||
# .idea/jarRepositories.xml
|
||||
# .idea/modules.xml
|
||||
# .idea/*.iml
|
||||
# .idea/modules
|
||||
# *.iml
|
||||
# *.ipr
|
||||
|
||||
# CMake
|
||||
cmake-build-*/
|
||||
|
||||
# Mongo Explorer plugin
|
||||
.idea/**/mongoSettings.xml
|
||||
|
||||
# File-based project format
|
||||
*.iws
|
||||
|
||||
# IntelliJ
|
||||
# out/
|
||||
|
||||
# mpeltonen/sbt-idea plugin
|
||||
.idea_modules/
|
||||
|
||||
# JIRA plugin
|
||||
atlassian-ide-plugin.xml
|
||||
|
||||
# Cursive Clojure plugin
|
||||
.idea/replstate.xml
|
||||
|
||||
# Crashlytics plugin (for Android Studio and IntelliJ)
|
||||
com_crashlytics_export_strings.xml
|
||||
crashlytics.properties
|
||||
crashlytics-build.properties
|
||||
fabric.properties
|
||||
|
||||
# Editor-based Rest Client
|
||||
.idea/httpRequests
|
||||
|
||||
# Android studio 3.1+ serialized cache file
|
||||
.idea/caches/build_file_checksums.ser
|
||||
|
||||
### Intellij+all Patch ###
|
||||
# Ignores the whole .idea folder and all .iml files
|
||||
# See https://github.com/joeblau/gitignore.io/issues/186 and https://github.com/joeblau/gitignore.io/issues/360
|
||||
|
||||
.idea/
|
||||
|
||||
# Reason: https://github.com/joeblau/gitignore.io/issues/186#issuecomment-249601023
|
||||
|
||||
*.iml
|
||||
modules.xml
|
||||
.idea/misc.xml
|
||||
*.ipr
|
||||
|
||||
# Sonarlint plugin
|
||||
.idea/sonarlint
|
||||
|
||||
### Java ###
|
||||
# Compiled class file
|
||||
*.class
|
||||
|
||||
# Log file
|
||||
*.log
|
||||
|
||||
# BlueJ files
|
||||
*.ctxt
|
||||
|
||||
# Mobile Tools for Java (J2ME)
|
||||
.mtj.tmp/
|
||||
|
||||
# Package Files #
|
||||
*.jar
|
||||
*.war
|
||||
*.nar
|
||||
*.ear
|
||||
*.zip
|
||||
*.tar.gz
|
||||
*.rar
|
||||
|
||||
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
|
||||
hs_err_pid*
|
||||
|
||||
### Gradle ###
|
||||
.gradle
|
||||
build/
|
||||
|
||||
# Ignore Gradle GUI config
|
||||
gradle-app.setting
|
||||
|
||||
# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
|
||||
!gradle-wrapper.jar
|
||||
|
||||
# Cache of project
|
||||
.gradletasknamecache
|
||||
|
||||
# # Work around https://youtrack.jetbrains.com/issue/IDEA-116898
|
||||
# gradle/wrapper/gradle-wrapper.properties
|
||||
|
||||
### Gradle Patch ###
|
||||
**/build/
|
||||
|
||||
# End of https://www.toptal.com/developers/gitignore/api/java,intellij+all,gradle
|
||||
|
||||
# Dumps of the meta-db
|
||||
data/*.db
|
||||
167
aws/s3/Readme.md
Normal file
167
aws/s3/Readme.md
Normal file
@@ -0,0 +1,167 @@
|
||||
<!-- PROJECT LOGO -->
|
||||
<br />
|
||||
<p align="center">
|
||||
<img src="images/logo.png" alt="Logo" width="200" height="200">
|
||||
<h3 align="center">Private File Upload</h3>
|
||||
|
||||
<p align="center">
|
||||
Build on Spring Cloud & AWS S3
|
||||
</p>
|
||||
</p>
|
||||
|
||||
|
||||
|
||||
<!-- TABLE OF CONTENTS -->
|
||||
## Table of Contents
|
||||
|
||||
* [About the Project](#about-the-project)
|
||||
* [Built With](#built-with)
|
||||
* [Getting Started](#getting-started)
|
||||
* [Prerequisites](#prerequisites)
|
||||
* [Installation](#installation)
|
||||
* [Usage](#usage)
|
||||
* [Contact](#contact)
|
||||
* [Acknowledgements](#acknowledgements)
|
||||
|
||||
|
||||
|
||||
<!-- ABOUT THE PROJECT -->
|
||||
## About The Project
|
||||
|
||||
<p align="center">
|
||||
<img src="images/usage.gif">
|
||||
</p>
|
||||
|
||||
|
||||
### Built With
|
||||
- Spring Boot
|
||||
- Spring Cloud
|
||||
- AWS S3
|
||||
- ☕️ & ❤️
|
||||
|
||||
<!-- GETTING STARTED -->
|
||||
## Getting Started
|
||||
|
||||
### Prerequisites
|
||||
|
||||
In order to run this application you should have an Amazon Web Services (AWS) account.
|
||||
|
||||
### Installation
|
||||
|
||||
1. Update the `application.yaml` settings value for `cloud.aws.credentials.profile-name` to the name of your local AWS-profile
|
||||
2. run `./gradlew bootRun`
|
||||
3. Open your browser at `localhost:8080`
|
||||
|
||||
<!-- USAGE EXAMPLES -->
|
||||
## Usage
|
||||
|
||||
The application wraps a lot of S3-related API calls in easy to use UI elements to
|
||||
|
||||
- create/delete buckets (mapped as "spaces" so we can give them non-unique names)
|
||||
- upload/delete files
|
||||
- make files public/private
|
||||
- create pre-signed URLs for files
|
||||
- create bucket policy rules to set the lifecycle duration for elements
|
||||
|
||||
More detailed instructions can be found here:
|
||||
|
||||
<details>
|
||||
<summary>🪣 Create a bucket</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Click on _New Space_
|
||||
3. Enter the name and click _Submit_
|
||||
4. A message should pop up to indicate success
|
||||
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>🗂 Upload a File</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Click on _Upload File_
|
||||
4. Pick our file, provide a name and click _Submit_
|
||||
5. A message should pop up to indicate success
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>🔎 List all Objects in a Bucket</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. You see a list of all objects stored in the bucket
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>🌐 Get an Object's URL</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Select _Download_ on the target object
|
||||
4. The object's URL shall be opened in a new tab
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>📢 Make an object public</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Select _Make Public_ on the target object
|
||||
4. A message should pop up to indicate success
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>🤫 Make an Object private</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Select _Make Private_ on the target object
|
||||
4. A message should pop up to indicate success
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>🔥 Delete an Object</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Select _Delete_ on the target object
|
||||
4. The list of objects should reload without the deleted one
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>☄️ Delete a Bucket</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Delete_ on the target Space/Bucket
|
||||
3. The list of buckets should reload without the deleted one
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>👾 Generate a pre-signed URL</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Details_ on the target Space/Bucket
|
||||
3. Select _Magic Link_ on the target object
|
||||
4. A message should pop up, containing a pre-signed URL for that object (which is valid for 15 minutes)
|
||||
</details>
|
||||
|
||||
<details>
|
||||
<summary>⏳ Set Expiration on Bucket</summary>
|
||||
|
||||
1. Navigate to the _Spaces_ section
|
||||
2. Select _Make Temporary_ on the target Space/Bucket
|
||||
3. Select _Delete_ on the target object
|
||||
4. A message should pop up to indicate success
|
||||
</details>
|
||||
|
||||
<!-- CONTACT -->
|
||||
## Contact
|
||||
|
||||
Joshua Görner - [jgoerner](https://www.linkedin.com/in/jgoerner/) - joshua.goerner[at]gmail.com
|
||||
|
||||
|
||||
<!-- ACKNOWLEDGEMENTS -->
|
||||
## Acknowledgements
|
||||
* [O. Drew](https://github.com/othneildrew/Best-README-Template) - nice GH Readme template
|
||||
* [Hatchful](https://hatchful.shopify.com/) - Easy Logo Generation
|
||||
41
aws/s3/build.gradle
Normal file
41
aws/s3/build.gradle
Normal file
@@ -0,0 +1,41 @@
|
||||
plugins {
|
||||
id 'org.springframework.boot' version '2.4.2'
|
||||
id 'io.spring.dependency-management' version '1.0.11.RELEASE'
|
||||
id 'java'
|
||||
}
|
||||
|
||||
group = 'io.jgoerner'
|
||||
version = '0.0.1-SNAPSHOT'
|
||||
sourceCompatibility = '11'
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
ext {
|
||||
// set('springCloudVersion', "Hoxton.SR9")
|
||||
set('springCloudVersion', "Finchley.SR1")
|
||||
}
|
||||
|
||||
|
||||
dependencies {
|
||||
annotationProcessor 'org.projectlombok:lombok'
|
||||
developmentOnly 'org.springframework.boot:spring-boot-devtools'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-web'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'
|
||||
implementation 'org.springframework.cloud:spring-cloud-starter-aws'
|
||||
implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
|
||||
implementation 'org.projectlombok:lombok:1.18.16'
|
||||
runtimeOnly 'com.h2database:h2'
|
||||
testImplementation 'org.springframework.boot:spring-boot-starter-test'
|
||||
}
|
||||
|
||||
dependencyManagement {
|
||||
imports {
|
||||
mavenBom "org.springframework.cloud:spring-cloud-dependencies:${springCloudVersion}"
|
||||
}
|
||||
}
|
||||
|
||||
test {
|
||||
useJUnitPlatform()
|
||||
}
|
||||
0
aws/s3/data/.gitkeep
Normal file
0
aws/s3/data/.gitkeep
Normal file
BIN
aws/s3/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
BIN
aws/s3/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
Binary file not shown.
6
aws/s3/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
6
aws/s3/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@@ -0,0 +1,6 @@
|
||||
#Mon Jan 04 22:38:53 CET 2021
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.7.1-all.zip
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
zipStorePath=wrapper/dists
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
185
aws/s3/gradlew
vendored
Executable file
185
aws/s3/gradlew
vendored
Executable file
@@ -0,0 +1,185 @@
|
||||
#!/usr/bin/env sh
|
||||
|
||||
#
|
||||
# Copyright 2015 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.
|
||||
#
|
||||
|
||||
##############################################################################
|
||||
##
|
||||
## 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='"-Xmx64m" "-Xms64m"'
|
||||
|
||||
# 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 or MSYS, switch paths to Windows format before running java
|
||||
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; 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=`expr $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"
|
||||
|
||||
exec "$JAVACMD" "$@"
|
||||
89
aws/s3/gradlew.bat
vendored
Normal file
89
aws/s3/gradlew.bat
vendored
Normal file
@@ -0,0 +1,89 @@
|
||||
@rem
|
||||
@rem Copyright 2015 the original author or authors.
|
||||
@rem
|
||||
@rem Licensed under the Apache License, Version 2.0 (the "License");
|
||||
@rem you may not use this file except in compliance with the License.
|
||||
@rem 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, software
|
||||
@rem distributed under the License is distributed on an "AS IS" BASIS,
|
||||
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
@rem See the License for the specific language governing permissions and
|
||||
@rem limitations under the License.
|
||||
@rem
|
||||
|
||||
@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 Resolve any "." and ".." in APP_HOME to make it shorter.
|
||||
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
|
||||
|
||||
@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="-Xmx64m" "-Xms64m"
|
||||
|
||||
@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 execute
|
||||
|
||||
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 execute
|
||||
|
||||
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
|
||||
|
||||
: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 %*
|
||||
|
||||
: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
|
||||
BIN
aws/s3/images/logo.png
Normal file
BIN
aws/s3/images/logo.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 60 KiB |
BIN
aws/s3/images/usage.gif
Normal file
BIN
aws/s3/images/usage.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 2.0 MiB |
1
aws/s3/settings.gradle
Normal file
1
aws/s3/settings.gradle
Normal file
@@ -0,0 +1 @@
|
||||
rootProject.name = 's3'
|
||||
38
aws/s3/src/main/java/io/jgoerner/s3/S3Application.java
Normal file
38
aws/s3/src/main/java/io/jgoerner/s3/S3Application.java
Normal file
@@ -0,0 +1,38 @@
|
||||
package io.jgoerner.s3;
|
||||
|
||||
import org.springframework.beans.factory.annotation.Value;
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
import org.springframework.boot.autoconfigure.thymeleaf.ThymeleafProperties;
|
||||
import org.springframework.cloud.aws.autoconfigure.context.ContextInstanceDataAutoConfiguration;
|
||||
import org.springframework.context.annotation.Bean;
|
||||
import org.thymeleaf.templateresolver.FileTemplateResolver;
|
||||
import org.thymeleaf.templateresolver.ITemplateResolver;
|
||||
|
||||
@SpringBootApplication(exclude = ContextInstanceDataAutoConfiguration.class)
|
||||
public class S3Application {
|
||||
|
||||
private final ThymeleafProperties properties;
|
||||
|
||||
@Value("${spring.thymeleaf.templates_root:}")
|
||||
private String templatesRoot;
|
||||
|
||||
public S3Application(ThymeleafProperties properties) {
|
||||
this.properties = properties;
|
||||
}
|
||||
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(S3Application.class, args);
|
||||
}
|
||||
|
||||
@Bean
|
||||
public ITemplateResolver defaultTemplateResolver() {
|
||||
FileTemplateResolver resolver = new FileTemplateResolver();
|
||||
resolver.setSuffix(properties.getSuffix());
|
||||
resolver.setPrefix(templatesRoot);
|
||||
resolver.setTemplateMode(properties.getMode());
|
||||
resolver.setCacheable(properties.isCache());
|
||||
return resolver;
|
||||
}
|
||||
|
||||
}
|
||||
128
aws/s3/src/main/java/io/jgoerner/s3/adapter/in/RestApi.java
Normal file
128
aws/s3/src/main/java/io/jgoerner/s3/adapter/in/RestApi.java
Normal file
@@ -0,0 +1,128 @@
|
||||
package io.jgoerner.s3.adapter.in;
|
||||
|
||||
import io.jgoerner.s3.application.port.in.object.*;
|
||||
import io.jgoerner.s3.application.port.in.space.*;
|
||||
import io.jgoerner.s3.domain.Object;
|
||||
import io.jgoerner.s3.domain.ObjectPartial;
|
||||
import io.jgoerner.s3.domain.Space;
|
||||
import io.jgoerner.s3.domain.SpacePartial;
|
||||
import lombok.SneakyThrows;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
import org.springframework.web.multipart.MultipartFile;
|
||||
|
||||
import java.net.URL;
|
||||
import java.util.List;
|
||||
import java.util.Optional;
|
||||
|
||||
@RestController
|
||||
@RequestMapping("api")
|
||||
@Log4j2
|
||||
public class RestApi {
|
||||
|
||||
private final CreateSpace spaceCreator;
|
||||
private final GetAllSpaces allSpaceGetter;
|
||||
private final RemoveSpace spaceRemover;
|
||||
private final GetAllObjects allObjectsInSpaceGetter;
|
||||
private final UploadObject objectUploader;
|
||||
private final UpdateObject objectUpdater;
|
||||
private final RemoveObject objectDeleter;
|
||||
private final ForceRemoveSpace forceSpaceRemover;
|
||||
private final CreateLink linkCreator;
|
||||
private final SetTTL ttlUpdater;
|
||||
private final RemoveTTL ttlRemover;
|
||||
|
||||
public RestApi(
|
||||
CreateSpace spaceCreator,
|
||||
GetAllSpaces allSpaceGetter,
|
||||
RemoveSpace spaceRemover,
|
||||
GetAllObjects allObjectsInSpaceGetter,
|
||||
UploadObject objectUploader,
|
||||
UpdateObject objectUpdater,
|
||||
RemoveObject objectDeleter,
|
||||
ForceRemoveSpace forceSpaceRemover,
|
||||
CreateLink linkCreator,
|
||||
SetTTL ttlUpdater,
|
||||
RemoveTTL ttlRemover) {
|
||||
this.spaceCreator = spaceCreator;
|
||||
this.allSpaceGetter = allSpaceGetter;
|
||||
this.spaceRemover = spaceRemover;
|
||||
this.allObjectsInSpaceGetter = allObjectsInSpaceGetter;
|
||||
this.objectUploader = objectUploader;
|
||||
this.objectUpdater = objectUpdater;
|
||||
this.objectDeleter = objectDeleter;
|
||||
this.forceSpaceRemover = forceSpaceRemover;
|
||||
this.linkCreator = linkCreator;
|
||||
this.ttlUpdater = ttlUpdater;
|
||||
this.ttlRemover = ttlRemover;
|
||||
}
|
||||
|
||||
@GetMapping("/space")
|
||||
List<Space> getSpaces() {
|
||||
return allSpaceGetter.getAll();
|
||||
}
|
||||
|
||||
@PostMapping("/space/{space}")
|
||||
Space postSpace(@PathVariable String space) {
|
||||
return spaceCreator.create(space);
|
||||
}
|
||||
|
||||
@DeleteMapping("/space/{space}")
|
||||
void deleteSpace(@PathVariable String space, @RequestParam Optional<Boolean> force) {
|
||||
log.info("Got the value " + force);
|
||||
force.ifPresentOrElse(
|
||||
value -> {
|
||||
if (value) {
|
||||
forceSpaceRemover.forceRemove(space);
|
||||
} else {
|
||||
spaceRemover.remove(space);
|
||||
}
|
||||
},
|
||||
() -> spaceRemover.remove(space));
|
||||
}
|
||||
|
||||
@GetMapping("/space/{space}/object")
|
||||
List<Object> getObjectsInSpace(@PathVariable String space) {
|
||||
return allObjectsInSpaceGetter.getAllObjects(space);
|
||||
}
|
||||
|
||||
@SneakyThrows
|
||||
@PostMapping("/space/{space}/object")
|
||||
Object postObject(
|
||||
@PathVariable String space,
|
||||
@RequestParam("file") MultipartFile file,
|
||||
@RequestParam(required = false, name = "name") String name) {
|
||||
var key = name != null ? name : file.getOriginalFilename();
|
||||
return objectUploader.upload(space, key, file.getInputStream());
|
||||
}
|
||||
|
||||
@PatchMapping("/space/{space}/object/{key}")
|
||||
Object patchObject(
|
||||
@PathVariable String space, @PathVariable String key, @RequestBody ObjectPartial body) {
|
||||
log.info("Got the partial " + body);
|
||||
return objectUpdater.update(space, key, body);
|
||||
}
|
||||
|
||||
@DeleteMapping("/space/{space}/object/{key}")
|
||||
void deleteObject(@PathVariable String space, @PathVariable String key) {
|
||||
objectDeleter.delete(space, key);
|
||||
}
|
||||
|
||||
@PostMapping("/space/{space}/object/{key}/url")
|
||||
URL createLink(
|
||||
@PathVariable String space,
|
||||
@PathVariable String key,
|
||||
@RequestParam(required = false, name = "duration", defaultValue = "300") Long duration) {
|
||||
return linkCreator.createLink(space, key, duration);
|
||||
}
|
||||
|
||||
@PatchMapping("/space/{space}")
|
||||
void patchSpace(@PathVariable String space, @RequestBody(required = false) SpacePartial body) {
|
||||
log.info("got " + body);
|
||||
if (body.getTtlInDays() > 1) {
|
||||
ttlUpdater.setTTL(space, body.getTtlInDays());
|
||||
} else {
|
||||
ttlRemover.removeTTL(space);
|
||||
}
|
||||
}
|
||||
}
|
||||
126
aws/s3/src/main/java/io/jgoerner/s3/adapter/in/View.java
Normal file
126
aws/s3/src/main/java/io/jgoerner/s3/adapter/in/View.java
Normal file
@@ -0,0 +1,126 @@
|
||||
package io.jgoerner.s3.adapter.in;
|
||||
|
||||
import io.jgoerner.s3.domain.ObjectPartial;
|
||||
import io.jgoerner.s3.domain.SpacePartial;
|
||||
import lombok.Data;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.springframework.stereotype.Controller;
|
||||
import org.springframework.ui.Model;
|
||||
import org.springframework.web.bind.annotation.*;
|
||||
import org.springframework.web.multipart.MultipartFile;
|
||||
import org.springframework.web.servlet.mvc.support.RedirectAttributes;
|
||||
|
||||
import java.util.Map;
|
||||
import java.util.Optional;
|
||||
|
||||
@Log4j2
|
||||
@Controller()
|
||||
public class View {
|
||||
|
||||
private final RestApi api;
|
||||
|
||||
public View(RestApi api) {
|
||||
this.api = api;
|
||||
}
|
||||
|
||||
@GetMapping("space")
|
||||
String space(Model model) {
|
||||
var spaces = api.getSpaces();
|
||||
model.addAttribute("spaces", spaces);
|
||||
return "space-overview";
|
||||
}
|
||||
|
||||
@GetMapping("space/{name}")
|
||||
String spaceDetail(@PathVariable String name, Model model) {
|
||||
model.addAllAttributes(Map.of("space", name, "objects", api.getObjectsInSpace(name)));
|
||||
return "space-detail";
|
||||
}
|
||||
|
||||
@GetMapping("space/{space}/make-public/{key}")
|
||||
String makePublic(
|
||||
@PathVariable String space, @PathVariable String key, RedirectAttributes redirectAttributes) {
|
||||
api.patchObject(space, key, new ObjectPartial(true));
|
||||
redirectAttributes.addFlashAttribute("success", "Made object public");
|
||||
return "redirect:/space/" + space;
|
||||
}
|
||||
|
||||
@GetMapping("space/{space}/make-private/{key}")
|
||||
String makePrivate(
|
||||
@PathVariable String space, @PathVariable String key, RedirectAttributes redirectAttributes) {
|
||||
api.patchObject(space, key, new ObjectPartial(false));
|
||||
redirectAttributes.addFlashAttribute("success", "Made object private");
|
||||
return "redirect:/space/" + space;
|
||||
}
|
||||
|
||||
@GetMapping("space/{space}/delete/{key}")
|
||||
String deleteObject(@PathVariable String space, @PathVariable String key) {
|
||||
api.deleteObject(space, key);
|
||||
return "redirect:/space/" + space;
|
||||
}
|
||||
|
||||
@GetMapping("/space/{space}/object-form")
|
||||
String objectNew(@PathVariable String space, Model model) {
|
||||
model.addAllAttributes(Map.of("space", space, "object", new ObjectForm()));
|
||||
return "object-form";
|
||||
}
|
||||
|
||||
@PostMapping("/space/{space}/object-form")
|
||||
String postObject(
|
||||
@PathVariable String space,
|
||||
@RequestParam("file") MultipartFile file,
|
||||
@ModelAttribute ObjectForm form,
|
||||
RedirectAttributes redirectAttributes) {
|
||||
api.postObject(space, file, form.getName());
|
||||
redirectAttributes.addFlashAttribute("success", "Successfully uploaded " + form.getName());
|
||||
return "redirect:/space/" + space;
|
||||
}
|
||||
|
||||
@GetMapping("/space/{space}/magic/{key}")
|
||||
String magicLink(
|
||||
@PathVariable String space, @PathVariable String key, RedirectAttributes redirectAttributes) {
|
||||
var link = api.createLink(space, key, 15L);
|
||||
redirectAttributes.addFlashAttribute("warn", link);
|
||||
return "redirect:/space/" + space;
|
||||
}
|
||||
|
||||
@GetMapping("space-form")
|
||||
String spaceNew(Model model) {
|
||||
model.addAttribute("space", new SpaceForm());
|
||||
return "space-form";
|
||||
}
|
||||
|
||||
@PostMapping("space-form")
|
||||
String newSpace(@ModelAttribute SpaceForm form, RedirectAttributes redirectAttributes) {
|
||||
api.postSpace(form.getName());
|
||||
redirectAttributes.addFlashAttribute("success", "Successfully created " + form.getName());
|
||||
return "redirect:space";
|
||||
}
|
||||
|
||||
@GetMapping("space/temporary/{name}")
|
||||
String makeTemporary(@PathVariable String name) {
|
||||
api.patchSpace(name, new SpacePartial(2));
|
||||
return "redirect:/space";
|
||||
}
|
||||
|
||||
@GetMapping("space/permanent/{name}")
|
||||
String makePermanent(@PathVariable String name) {
|
||||
api.patchSpace(name, new SpacePartial(-1));
|
||||
return "redirect:/space";
|
||||
}
|
||||
|
||||
@GetMapping("space/delete/{name}")
|
||||
String delete(@PathVariable String name) {
|
||||
api.deleteSpace(name, Optional.of(true));
|
||||
return "redirect:/space";
|
||||
}
|
||||
}
|
||||
|
||||
@Data
|
||||
class SpaceForm {
|
||||
String name;
|
||||
}
|
||||
|
||||
@Data
|
||||
class ObjectForm {
|
||||
String name;
|
||||
}
|
||||
@@ -0,0 +1,20 @@
|
||||
package io.jgoerner.s3.adapter.out.h2;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Getter;
|
||||
import lombok.NoArgsConstructor;
|
||||
import lombok.Setter;
|
||||
|
||||
import javax.persistence.Entity;
|
||||
import javax.persistence.Id;
|
||||
|
||||
@Entity
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
@Getter
|
||||
@Setter
|
||||
public class SpaceEntity {
|
||||
@Id private String name;
|
||||
private String bucket;
|
||||
private Integer ttl;
|
||||
}
|
||||
@@ -0,0 +1,69 @@
|
||||
package io.jgoerner.s3.adapter.out.h2;
|
||||
|
||||
import io.jgoerner.s3.application.port.out.space.*;
|
||||
import io.jgoerner.s3.domain.Space;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.springframework.stereotype.Component;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
@Component
|
||||
@Log4j2
|
||||
public class SpacePersistenceHandler
|
||||
implements CheckSpaceExistence,
|
||||
SaveSpace,
|
||||
RetrieveAllSpaces,
|
||||
ResolveSpaceName,
|
||||
DeleteSpace,
|
||||
RetrieveSpaceByName {
|
||||
|
||||
private final SpaceRepository spaceRepository;
|
||||
|
||||
public SpacePersistenceHandler(SpaceRepository spaceRepository) {
|
||||
this.spaceRepository = spaceRepository;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean doesExist(String name) {
|
||||
return this.spaceRepository.findById(name).isPresent();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Space save(Space name) {
|
||||
this.spaceRepository.save(SpacePersistenceHandler.mapPojoToJpa(name));
|
||||
return name;
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<Space> findAll() {
|
||||
return spaceRepository.findAll().stream()
|
||||
.map(SpacePersistenceHandler::mapJpaToPojo)
|
||||
.collect(Collectors.toList());
|
||||
}
|
||||
|
||||
private static SpaceEntity mapPojoToJpa(Space space) {
|
||||
return new SpaceEntity(space.getName(), space.getBucket(), space.getTtl());
|
||||
}
|
||||
|
||||
private static Space mapJpaToPojo(SpaceEntity entity) {
|
||||
return new Space(entity.getName(), entity.getBucket(), entity.getTtl());
|
||||
}
|
||||
|
||||
@Override
|
||||
public String resolve(String name) {
|
||||
var bucket = spaceRepository.findById(name).get().getBucket();
|
||||
log.info("Space " + name + " was resolved to " + bucket);
|
||||
return bucket;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void delete(String name) {
|
||||
spaceRepository.deleteById(name);
|
||||
}
|
||||
|
||||
@Override
|
||||
public Space retrieveByName(String name) {
|
||||
return mapJpaToPojo(spaceRepository.findById(name).orElseThrow());
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
package io.jgoerner.s3.adapter.out.h2;
|
||||
|
||||
import org.springframework.data.jpa.repository.JpaRepository;
|
||||
import org.springframework.stereotype.Repository;
|
||||
|
||||
@Repository
|
||||
public interface SpaceRepository extends JpaRepository<SpaceEntity, String> {}
|
||||
@@ -0,0 +1,143 @@
|
||||
package io.jgoerner.s3.adapter.out.s3;
|
||||
|
||||
import com.amazonaws.services.s3.AmazonS3Client;
|
||||
import com.amazonaws.services.s3.model.*;
|
||||
import com.amazonaws.services.s3.model.lifecycle.LifecycleFilter;
|
||||
import com.amazonaws.waiters.WaiterParameters;
|
||||
import io.jgoerner.s3.application.port.out.bucket.*;
|
||||
import io.jgoerner.s3.application.port.out.object.*;
|
||||
import io.jgoerner.s3.domain.Object;
|
||||
import lombok.extern.log4j.Log4j2;
|
||||
import org.springframework.stereotype.Repository;
|
||||
|
||||
import java.io.InputStream;
|
||||
import java.net.URL;
|
||||
import java.util.Date;
|
||||
import java.util.List;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
@Repository
|
||||
@Log4j2
|
||||
public class S3Repository
|
||||
implements CreateBucket,
|
||||
DeleteBucket,
|
||||
ListObjects,
|
||||
SaveObject,
|
||||
MakeObjectPublic,
|
||||
MakeObjectPrivate,
|
||||
DeleteObject,
|
||||
CreatePresignedUrl,
|
||||
SetVisibilityInObjectLifecycle,
|
||||
RemoveVisibilityInObjectLifecycle {
|
||||
|
||||
private final AmazonS3Client s3Client;
|
||||
|
||||
public S3Repository(AmazonS3Client s3Client) {
|
||||
this.s3Client = s3Client;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void create(String bucket) {
|
||||
// send bucket creation request
|
||||
s3Client.createBucket(bucket);
|
||||
log.info("Request to create " + bucket + " sent");
|
||||
|
||||
// assure that bucket is available
|
||||
s3Client.waiters().bucketExists().run(new WaiterParameters<>(new HeadBucketRequest(bucket)));
|
||||
log.info("Bucket " + bucket + " is ready");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void delete(String bucket) {
|
||||
// send deletion request
|
||||
s3Client.deleteBucket(bucket);
|
||||
log.info("Request to delete " + bucket + " sent");
|
||||
|
||||
// assure bucket is deleted
|
||||
s3Client.waiters().bucketNotExists().run(new WaiterParameters(new HeadBucketRequest(bucket)));
|
||||
log.info("Bucket " + bucket + " is deleted");
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<Object> listObjectsInBucket(String bucket) {
|
||||
var items =
|
||||
s3Client.listObjectsV2(bucket).getObjectSummaries().stream()
|
||||
.parallel()
|
||||
.map(S3ObjectSummary::getKey)
|
||||
.map(key -> mapS3ToObject(bucket, key))
|
||||
.collect(Collectors.toList());
|
||||
|
||||
log.info("Found " + items.size() + " objects in the bucket " + bucket);
|
||||
return items;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object safe(String bucket, String key, String name, InputStream payload) {
|
||||
var metadata = new ObjectMetadata();
|
||||
metadata.addUserMetadata("name", name);
|
||||
s3Client.putObject(bucket, key, payload, metadata);
|
||||
log.info("Sent the request");
|
||||
return Object.builder().name(name).key(key).url(s3Client.getUrl(bucket, key)).build();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void makePublic(String bucket, String key) {
|
||||
s3Client.setObjectAcl(bucket, key, CannedAccessControlList.PublicRead);
|
||||
log.info("Sent request to make object in bucket " + bucket + " with key " + key + " public");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void makePrivate(String bucket, String key) {
|
||||
s3Client.setObjectAcl(bucket, key, CannedAccessControlList.BucketOwnerFullControl);
|
||||
log.info("Sent request to make object in bucket " + bucket + " with key " + key + " private");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void delete(String bucket, String key) {
|
||||
s3Client.deleteObject(bucket, key);
|
||||
log.info("Sent request to delete file with key " + key + " in bucket " + bucket);
|
||||
}
|
||||
|
||||
@Override
|
||||
public URL createURL(String bucket, String key, Long duration) {
|
||||
var date = new Date(new Date().getTime() + duration * 1000); // 1 s * 1000 ms/s
|
||||
var url = s3Client.generatePresignedUrl(bucket, key, date);
|
||||
log.info("Generated the signature " + url);
|
||||
return url;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setVisibility(String bucket, Integer ttlInDays) {
|
||||
s3Client.setBucketLifecycleConfiguration(
|
||||
bucket,
|
||||
new BucketLifecycleConfiguration()
|
||||
.withRules(
|
||||
new BucketLifecycleConfiguration.Rule()
|
||||
.withId("custom-expiration-id")
|
||||
.withFilter(new LifecycleFilter())
|
||||
.withStatus(BucketLifecycleConfiguration.ENABLED)
|
||||
.withExpirationInDays(ttlInDays)));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void removeVisibility(String bucket) {
|
||||
s3Client.deleteBucketLifecycleConfiguration(bucket);
|
||||
}
|
||||
|
||||
private Object mapS3ToObject(String bucket, String key) {
|
||||
|
||||
return Object.builder()
|
||||
.name(s3Client.getObjectMetadata(bucket, key).getUserMetaDataOf("name"))
|
||||
.key(key)
|
||||
.url(s3Client.getUrl(bucket, key))
|
||||
.isPublic(
|
||||
s3Client.getObjectAcl(bucket, key).getGrantsAsList().stream()
|
||||
.anyMatch(grant -> grant.equals(S3Repository.publicObjectReadGrant())))
|
||||
.build();
|
||||
}
|
||||
|
||||
private static Grant publicObjectReadGrant() {
|
||||
return new Grant(
|
||||
GroupGrantee.parseGroupGrantee(GroupGrantee.AllUsers.getIdentifier()), Permission.Read);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
package io.jgoerner.s3.application.port.in.object;
|
||||
|
||||
import java.net.URL;
|
||||
|
||||
public interface CreateLink {
|
||||
URL createLink(String space, String key, Long duration);
|
||||
}
|
||||
@@ -0,0 +1,9 @@
|
||||
package io.jgoerner.s3.application.port.in.object;
|
||||
|
||||
import io.jgoerner.s3.domain.Object;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public interface GetAllObjects {
|
||||
List<Object> getAllObjects(String space);
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user