docs/code-samples/serving/hello-world/helloworld-java-spring/README.md

231 lines
7.2 KiB
Markdown

# Hello World - Spring Boot Java
This guide describes the steps required to create the `helloworld-java-spring` sample app and deploy it to your cluster.
The sample app reads a `TARGET` environment variable, and prints `Hello ${TARGET}!`. If `TARGET` is not specified, `World` is used as the default value.
You can also download a working copy of the sample, by running the following commands:
```bash
git clone https://github.com/knative/docs.git knative-docs
cd knative-docs/code-samples/serving/hello-world/helloworld-java-spring
```
## Prerequisites
- A Kubernetes cluster with Knative installed and DNS configured. See
[Install Knative Serving](https://knative.dev/docs/install/serving/install-serving-with-yaml).
- [Docker](https://www.docker.com) installed and running on your local machine, and a Docker Hub account configured.
- Optional. The Knative CLI client [`kn`](https://github.com/knative/client/releases) can be used to simplify the deployment. Alternatively, you can use `kubectl` to apply YAML resource files.
## Building the sample app
1. From the console, create a new, empty web project by using the `curl` and `unzip`
commands:
```bash
curl https://start.spring.io/starter.zip \
-d dependencies=web \
-d name=helloworld \
-d artifactId=helloworld \
-o helloworld.zip
unzip helloworld.zip
```
If you don't have `curl` installed, you can accomplish the same by visiting the
[Spring Initializr](https://start.spring.io/) page. Specify Artifact as
`helloworld` and add the `Web` dependency. Then click **Generate Project**,
download and unzip the sample archive.
1. Update the `SpringBootApplication` class in
`src/main/java/com/example/helloworld/HelloworldApplication.java` by adding a
`@RestController` to handle the "/" mapping and also add a `@Value` field to
provide the `TARGET` environment variable:
```java
package com.example.helloworld;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@SpringBootApplication
public class HelloworldApplication {
@Value("${TARGET:World}")
String target;
@RestController
class HelloworldController {
@GetMapping("/")
String hello() {
return "Hello " + target + "!";
}
}
public static void main(String[] args) {
SpringApplication.run(HelloworldApplication.class, args);
}
}
```
1. Run the application locally:
```bash
mvn wrapper:wrapper
./mvnw package && java -jar target/helloworld-0.0.1-SNAPSHOT.jar
```
Go to `http://localhost:8080/` to see your `Hello World!` message.
1. In your project directory, create a file named `Dockerfile` and copy the following code block into it:
```docker
# Use the official maven/Java 8 image to create a build artifact: https://hub.docker.com/_/maven
FROM maven:3.5-jdk-8-alpine AS builder
# Copy local code to the container image.
WORKDIR /app
COPY pom.xml .
COPY src ./src
# Build a release artifact.
RUN mvn package -DskipTests
# Use the Official OpenJDK image for a lean production stage of our multi-stage build.
# https://hub.docker.com/_/openjdk
# https://docs.docker.com/develop/develop-images/multistage-build/#use-multi-stage-builds
FROM openjdk:8-jre-alpine
# Copy the jar to the production image from the builder stage.
COPY --from=builder /app/target/helloworld-*.jar /helloworld.jar
# Run the web service on container startup.
CMD ["java", "-Djava.security.egd=file:/dev/./urandom", "-jar", "/helloworld.jar"]
```
For detailed instructions on dockerizing a Spring Boot app, see [Spring Boot with Docker](https://spring.io/guides/gs/spring-boot-docker/).
For additional information on multi-stage docker builds for Java see [Creating Smaller Java Image using Docker Multi-stage Build](http://blog.arungupta.me/smaller-java-image-docker-multi-stage-build/).
1. Use Docker to build the sample code into a container, then push the container to the Docker registry:
```bash
# Build and push the container on your local machine.
docker buildx build --platform linux/arm64,linux/amd64 -t "{username}/helloworld-java-spring" --push .
```
Where `{username}` is your Docker Hub username.
## Deploying the app
After the build has completed and the container is pushed to Docker Hub, you can deploy the app into your cluster.
During the creation of a Service, Knative performs the following steps:
- Create a new immutable revision for this version of the app.
- Network programming to create a Route, ingress, Service, and load balancer for your app.
- Automatically scale your pods up and down, including scaling down to zero active pods.
Choose one of the following methods to deploy the app:
### yaml
1. Create a new file named `service.yaml` and copy the following service definition
into the file:
```yaml
apiVersion: serving.knative.dev/v1
kind: Service
metadata:
name: helloworld-java-spring
namespace: default
spec:
template:
spec:
containers:
- image: docker.io/{username}/helloworld-java-spring
env:
- name: TARGET
value: "Spring Boot Sample v1"
```
Where `{username}` is your Docker Hub username.
**Note:** Ensure that the container image value in `service.yaml` matches the container you built in the previous step.
1. Apply the YAML file by running the command:
```bash
kubectl apply -f service.yaml
```
### kn
1. With `kn` you can deploy the service with
```bash
kn service create helloworld-java-spring --image=docker.io/{username}/helloworld-java-spring --env TARGET="Spring Boot Sample v1"
```
This will wait until your service is deployed and ready, and ultimately it will print the URL through which you can access the service.
## Verification
1. Find the domain URL for your service:
- For kubectl, run:
```bash
kubectl get ksvc helloworld-java-spring --output=custom-columns=NAME:.metadata.name,URL:.status.url
```
Example:
```bash
NAME URL
helloworld-java-spring http://helloworld-java-spring.default.1.2.3.4.xip.io
```
- For kn, run:
```bash
kn service describe helloworld-java-spring -o url
```
Example:
```bash
http://helloworld-java-spring.default.1.2.3.4.xip.io
```
1. Make a request to your app and observe the result. Replace
the following URL with the URL returned in the previous command.
Example:
```bash
curl http://helloworld-java-spring.default.1.2.3.4.sslip.io
Hello Spring Boot Sample v1!
# Even easier with kn:
curl $(kn service describe helloworld-java-spring -o url)
```
**Tip:** Add `-v` option to get more detail if the `curl` command fails.
## Deleting the app
To remove the sample app from your cluster, delete the service:
### kubectl
```bash
kubectl delete -f service.yaml
```
### kn
```bash
kn service delete helloworld-java-spring
```