mirror of
https://github.com/stefanprodan/podinfo.git
synced 2026-04-09 12:36:50 +00:00
Compare commits
331 Commits
6.1.1
...
test-sourc
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f6f8b33112 | ||
|
|
28a6cea67f | ||
|
|
e86405a867 | ||
|
|
a51de59edb | ||
|
|
1ff6fff334 | ||
|
|
98e8cd39d4 | ||
|
|
65b9e92db3 | ||
|
|
c6536c75ce | ||
|
|
3c4910d1c7 | ||
|
|
c831679d1e | ||
|
|
94a9f99f79 | ||
|
|
86ac641693 | ||
|
|
cdd09cdd3d | ||
|
|
0b8a7dace7 | ||
|
|
e50f88a43d | ||
|
|
67e4628d95 | ||
|
|
73f5e936c8 | ||
|
|
0e26c3b934 | ||
|
|
dc39bd9a08 | ||
|
|
c23d57a4e9 | ||
|
|
85d22b2172 | ||
|
|
ff906f2242 | ||
|
|
2dbf735c46 | ||
|
|
1318243ec9 | ||
|
|
12e7f14ff0 | ||
|
|
fb3b01be30 | ||
|
|
be955e76b0 | ||
|
|
3e79d79447 | ||
|
|
cd555cf439 | ||
|
|
2927336ecd | ||
|
|
8f5425b6d5 | ||
|
|
1cd88028c7 | ||
|
|
ca101e6728 | ||
|
|
c8419e386a | ||
|
|
9356c1c0c4 | ||
|
|
60b5e4c4fe | ||
|
|
322b71c1e5 | ||
|
|
b3396adb98 | ||
|
|
c6e70a7aa0 | ||
|
|
6810566623 | ||
|
|
8efcd73510 | ||
|
|
eb81ed6ed2 | ||
|
|
916f1a298a | ||
|
|
7cc399463c | ||
|
|
78755636d0 | ||
|
|
b99bf8c252 | ||
|
|
e7928cfbc7 | ||
|
|
dc5d3d559e | ||
|
|
71f618320e | ||
|
|
c4e601b4cb | ||
|
|
6b7aab8a10 | ||
|
|
b7d3d71d8f | ||
|
|
abcd272293 | ||
|
|
4af7854aa2 | ||
|
|
fc1e4a48ed | ||
|
|
08238eada7 | ||
|
|
892a66ea93 | ||
|
|
0b1481aa8e | ||
|
|
ff32a1fc4b | ||
|
|
3de84d2360 | ||
|
|
ba6f4ffd7c | ||
|
|
6d4405a1ef | ||
|
|
752950cb4f | ||
|
|
b10c3067c8 | ||
|
|
85cd1c46d4 | ||
|
|
2687a13c75 | ||
|
|
5fb6597929 | ||
|
|
1fbdd9420f | ||
|
|
30cd3b27d7 | ||
|
|
1d7de0bb82 | ||
|
|
c52654c59e | ||
|
|
1a6838a4a2 | ||
|
|
b0c487c6b2 | ||
|
|
b28069ac51 | ||
|
|
d8a136cf74 | ||
|
|
614f74f6df | ||
|
|
0c4f327390 | ||
|
|
d791dedb24 | ||
|
|
220d4e909c | ||
|
|
eba7fe186e | ||
|
|
9108833214 | ||
|
|
386ceb09be | ||
|
|
f350624047 | ||
|
|
37ba3e854f | ||
|
|
70335812c6 | ||
|
|
8d010c498e | ||
|
|
8b3079a417 | ||
|
|
37fa020bcd | ||
|
|
d879d0f4fb | ||
|
|
16191504d1 | ||
|
|
d042732a44 | ||
|
|
649864583b | ||
|
|
c07eb64558 | ||
|
|
44942884c3 | ||
|
|
d562a2a82a | ||
|
|
d7c1bf015c | ||
|
|
cdec0786ef | ||
|
|
e6d611e1e2 | ||
|
|
36bea810ef | ||
|
|
50047dab3a | ||
|
|
2b936e6700 | ||
|
|
55e4e51eba | ||
|
|
47090ad9e1 | ||
|
|
6a0bbda8a5 | ||
|
|
357009a863 | ||
|
|
0f98770296 | ||
|
|
f9032836a6 | ||
|
|
5368c3fe10 | ||
|
|
b1207aa9b1 | ||
|
|
c7f9b521fa | ||
|
|
24405a5a5d | ||
|
|
5195b158fc | ||
|
|
532db405f8 | ||
|
|
2251bee699 | ||
|
|
8535efccb7 | ||
|
|
e008d1f261 | ||
|
|
22097353d2 | ||
|
|
c305843105 | ||
|
|
0d2c428859 | ||
|
|
ecaa7cf4d3 | ||
|
|
8447b6985b | ||
|
|
9371d6d153 | ||
|
|
20b8c1043c | ||
|
|
2d80c7a22d | ||
|
|
dc830d02a6 | ||
|
|
badf3271a1 | ||
|
|
9f9c2f3245 | ||
|
|
33dac1ba40 | ||
|
|
1cf8b8aeef | ||
|
|
22fb1c3d34 | ||
|
|
adf8157da6 | ||
|
|
03f8ad0251 | ||
|
|
101e371e96 | ||
|
|
53c9f3ad9b | ||
|
|
a69f0282fd | ||
|
|
174d183056 | ||
|
|
7bb64e7567 | ||
|
|
49a4b31d53 | ||
|
|
c29e013a33 | ||
|
|
45ecda63bb | ||
|
|
4868f430c7 | ||
|
|
ca9c6bb4f8 | ||
|
|
86f3b1a57c | ||
|
|
b22dd96a54 | ||
|
|
5aaf95849e | ||
|
|
e197eca420 | ||
|
|
73fcdbe4a6 | ||
|
|
074d0f9ff2 | ||
|
|
d9bc6301e9 | ||
|
|
a0e323e331 | ||
|
|
1ee349fa17 | ||
|
|
0f526c3cd4 | ||
|
|
021c55fed9 | ||
|
|
bb2408d17d | ||
|
|
5eb3cafd6a | ||
|
|
df0f8ba885 | ||
|
|
19a59d96f1 | ||
|
|
401461595a | ||
|
|
bd77584ade | ||
|
|
87e0dbaa7e | ||
|
|
c5494104a1 | ||
|
|
74c60a927c | ||
|
|
ecdf07c4d5 | ||
|
|
ff29c549ff | ||
|
|
fa75fc0520 | ||
|
|
0bc496456d | ||
|
|
398c543171 | ||
|
|
a54dc2a9c7 | ||
|
|
bfa42afa1f | ||
|
|
590987704e | ||
|
|
d561182076 | ||
|
|
72bd6faf35 | ||
|
|
2cbe0fcdff | ||
|
|
87e594b109 | ||
|
|
7ec9e6c84a | ||
|
|
8183d0d5fc | ||
|
|
aa27416651 | ||
|
|
b0594a85b9 | ||
|
|
e816d1b5bc | ||
|
|
6316e213d1 | ||
|
|
a1b112f4e1 | ||
|
|
1495fd888e | ||
|
|
dfc4a6d37e | ||
|
|
aaa47e535f | ||
|
|
0278e11a05 | ||
|
|
12ceae475f | ||
|
|
4892983fd1 | ||
|
|
bcf492e92b | ||
|
|
a54550e439 | ||
|
|
29dd482f49 | ||
|
|
3a7d4d1544 | ||
|
|
c14b116dea | ||
|
|
12c078938d | ||
|
|
dd3869b1a1 | ||
|
|
45cfe3abc2 | ||
|
|
fcf573111b | ||
|
|
cadabcc6a5 | ||
|
|
9dfb676083 | ||
|
|
e06a5517da | ||
|
|
fedab0de38 | ||
|
|
7d13025a35 | ||
|
|
7280e43cbf | ||
|
|
3ef0b4cd09 | ||
|
|
073f1ec5af | ||
|
|
1e0307c759 | ||
|
|
d4d75c2fbf | ||
|
|
2a6533c68a | ||
|
|
0647aea75b | ||
|
|
8c258bb1d8 | ||
|
|
58726f0bd2 | ||
|
|
bc08542ed3 | ||
|
|
bbce3f3f67 | ||
|
|
67e2c98a60 | ||
|
|
938b00be6d | ||
|
|
e6c7657155 | ||
|
|
d75e8d7838 | ||
|
|
74d6532429 | ||
|
|
8187f79475 | ||
|
|
2b6f4f0a7d | ||
|
|
3a4a99697b | ||
|
|
1abc44f0d8 | ||
|
|
3d798af827 | ||
|
|
f8f8073946 | ||
|
|
c8c7a6d1bb | ||
|
|
eac008b339 | ||
|
|
d2227a4204 | ||
|
|
ae3fe3da98 | ||
|
|
42fdaf8e7a | ||
|
|
3e2d907993 | ||
|
|
21136b6405 | ||
|
|
e8c388a3fd | ||
|
|
abc38e1bff | ||
|
|
bf4a3140fe | ||
|
|
de2dd687cb | ||
|
|
f7a9563986 | ||
|
|
a699fffe7b | ||
|
|
24e5de8934 | ||
|
|
298c1ae941 | ||
|
|
fdd0a0b7da | ||
|
|
8bab17843c | ||
|
|
34c5ab57b6 | ||
|
|
0f9c989b68 | ||
|
|
e2e85a9604 | ||
|
|
b687d3c76f | ||
|
|
dbbb415194 | ||
|
|
1a89d81ebb | ||
|
|
b39526ebe8 | ||
|
|
607303dca9 | ||
|
|
3053e634f9 | ||
|
|
4f1e56ae83 | ||
|
|
f0590a03e0 | ||
|
|
aa815625d9 | ||
|
|
8615cb75d9 | ||
|
|
b23ebb15cb | ||
|
|
dcb5b13023 | ||
|
|
71869089fa | ||
|
|
1cf228c67b | ||
|
|
b6e81a931b | ||
|
|
744597a481 | ||
|
|
389c86ee93 | ||
|
|
34db5fa463 | ||
|
|
0d62402ae9 | ||
|
|
e40d32ba87 | ||
|
|
3879b59f43 | ||
|
|
44157ecd84 | ||
|
|
bfa8d8032f | ||
|
|
b1251214f6 | ||
|
|
f1168c4946 | ||
|
|
013343a232 | ||
|
|
d460863f3b | ||
|
|
25a1e26159 | ||
|
|
b39afea117 | ||
|
|
6d11ef9baf | ||
|
|
baf128d856 | ||
|
|
79f8138328 | ||
|
|
ceed4e7870 | ||
|
|
bfce2199e8 | ||
|
|
d55bb8eabd | ||
|
|
5fb056ebcb | ||
|
|
35b9c9f946 | ||
|
|
74e0aeeff7 | ||
|
|
bbb081b0e1 | ||
|
|
c16318bb85 | ||
|
|
86d5fe86e4 | ||
|
|
b3b00fe354 | ||
|
|
a7bcfaf9b3 | ||
|
|
1d4c534728 | ||
|
|
f2e0aa154d | ||
|
|
6d5b3d254a | ||
|
|
9b9f11da95 | ||
|
|
1a55e30bcf | ||
|
|
394c40e3ff | ||
|
|
b76b1a38c9 | ||
|
|
2eb17d80c8 | ||
|
|
678a42ce34 | ||
|
|
2da59980fe | ||
|
|
8697f091f3 | ||
|
|
4d2cf65260 | ||
|
|
116a378991 | ||
|
|
450796ddb2 | ||
|
|
cb8c1fcec1 | ||
|
|
37da8d1c74 | ||
|
|
e55ebd258d | ||
|
|
6b869d1a18 | ||
|
|
dea973d614 | ||
|
|
f4199ab8bc | ||
|
|
19603ddfc1 | ||
|
|
bf09377bfd | ||
|
|
075712dd73 | ||
|
|
07dd9a3c3e | ||
|
|
63ac69ea69 | ||
|
|
3db382d2c9 | ||
|
|
9f88a0e940 | ||
|
|
c6a2c90497 | ||
|
|
54908f7d51 | ||
|
|
36bf90b008 | ||
|
|
dd9020c8b2 | ||
|
|
51009591a5 | ||
|
|
2b8c71ba78 | ||
|
|
203f7e1bf0 | ||
|
|
8179263f52 | ||
|
|
b26a34b5b6 | ||
|
|
cd7a0fb18e | ||
|
|
c1fd17e50a | ||
|
|
f98267009e | ||
|
|
7d0203196a | ||
|
|
673966bae4 | ||
|
|
9265828c4f | ||
|
|
0f68b60870 | ||
|
|
217a27ce02 | ||
|
|
fc172b0e7c |
@@ -1,9 +1,10 @@
|
|||||||
# Podinfo signed releases
|
# Podinfo signed releases
|
||||||
|
|
||||||
Podinfo deployment manifests are published to GitHub Container Registry as OCI artifacts
|
Podinfo release assets (container image, Helm chart, Flux artifact, Timoni module)
|
||||||
and are signed using [cosign](https://github.com/sigstore/cosign).
|
are published to GitHub Container Registry and are signed with
|
||||||
|
[Cosign v2](https://github.com/sigstore/cosign) keyless & GitHub Actions OIDC.
|
||||||
|
|
||||||
## Verify the artifacts with cosign
|
## Verify podinfo with cosign
|
||||||
|
|
||||||
Install the [cosign](https://github.com/sigstore/cosign) CLI:
|
Install the [cosign](https://github.com/sigstore/cosign) CLI:
|
||||||
|
|
||||||
@@ -11,29 +12,50 @@ Install the [cosign](https://github.com/sigstore/cosign) CLI:
|
|||||||
brew install sigstore/tap/cosign
|
brew install sigstore/tap/cosign
|
||||||
```
|
```
|
||||||
|
|
||||||
Verify a podinfo release with cosign CLI:
|
### Container image
|
||||||
|
|
||||||
|
Verify the podinfo container image hosted on GHCR:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
cosign verify -key https://raw.githubusercontent.com/stefanprodan/podinfo/master/cosign/cosign.pub \
|
cosign verify ghcr.io/stefanprodan/podinfo:6.5.0 \
|
||||||
ghcr.io/stefanprodan/podinfo-deploy:latest
|
--certificate-identity-regexp="^https://github.com/stefanprodan/podinfo.*$" \
|
||||||
|
--certificate-oidc-issuer=https://token.actions.githubusercontent.com
|
||||||
```
|
```
|
||||||
|
|
||||||
## Download the artifacts with crane
|
Verify the podinfo container image hosted on Docker Hub:
|
||||||
|
|
||||||
Install the [crane](https://github.com/google/go-containerregistry/tree/main/cmd/crane) CLI:
|
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
brew install crane
|
cosign verify docker.io/stefanprodan/podinfo:6.5.0 \
|
||||||
|
--certificate-identity-regexp="^https://github.com/stefanprodan/podinfo.*$" \
|
||||||
|
--certificate-oidc-issuer=https://token.actions.githubusercontent.com
|
||||||
```
|
```
|
||||||
|
|
||||||
Download the podinfo deployment manifests with crane CLI:
|
### Helm chart
|
||||||
|
|
||||||
```console
|
Verify the podinfo [Helm](https://helm.sh) chart hosted on GHCR:
|
||||||
$ crane export ghcr.io/stefanprodan/podinfo-deploy:latest -| tar -xf -
|
|
||||||
|
|
||||||
$ ls -1
|
```sh
|
||||||
deployment.yaml
|
cosign verify ghcr.io/stefanprodan/charts/podinfo:6.5.0 \
|
||||||
hpa.yaml
|
--certificate-identity-regexp="^https://github.com/stefanprodan/podinfo.*$" \
|
||||||
kustomization.yaml
|
--certificate-oidc-issuer=https://token.actions.githubusercontent.com
|
||||||
service.yaml
|
```
|
||||||
|
|
||||||
|
### Flux artifact
|
||||||
|
|
||||||
|
Verify the podinfo [Flux](https://fluxcd.io) artifact hosted on GHCR:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cosign verify ghcr.io/stefanprodan/manifests/podinfo:6.5.0 \
|
||||||
|
--certificate-identity-regexp="^https://github.com/stefanprodan/podinfo.*$" \
|
||||||
|
--certificate-oidc-issuer=https://token.actions.githubusercontent.com
|
||||||
|
```
|
||||||
|
|
||||||
|
### Timoni module
|
||||||
|
|
||||||
|
Verify the podinfo [Timoni](https://timoni.sh) module hosted on GHCR:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cosign verify ghcr.io/stefanprodan/modules/podinfo:6.5.0 \
|
||||||
|
--certificate-identity-regexp="^https://github.com/stefanprodan/podinfo.*$" \
|
||||||
|
--certificate-oidc-issuer=https://token.actions.githubusercontent.com
|
||||||
```
|
```
|
||||||
|
|||||||
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
timoni/podinfo/cue.mod/** linguist-vendored
|
||||||
1
.github/FUNDING.yml
vendored
Normal file
1
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
github: stefanprodan
|
||||||
33
.github/actions/helm/action.yml
vendored
33
.github/actions/helm/action.yml
vendored
@@ -1,33 +0,0 @@
|
|||||||
name: Setup Helm CLI
|
|
||||||
description: A GitHub Action for running Helm commands
|
|
||||||
author: Stefan Prodan
|
|
||||||
branding:
|
|
||||||
color: blue
|
|
||||||
icon: command
|
|
||||||
inputs:
|
|
||||||
version:
|
|
||||||
description: "Helm version"
|
|
||||||
required: true
|
|
||||||
runs:
|
|
||||||
using: composite
|
|
||||||
steps:
|
|
||||||
- name: "Download helm binary to tmp"
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
VERSION=${{ inputs.version }}
|
|
||||||
BIN_URL="https://get.helm.sh/helm-v${VERSION}-linux-amd64.tar.gz"
|
|
||||||
curl -sL ${BIN_URL} -o /tmp/helm.tar.gz
|
|
||||||
mkdir -p /tmp/helm
|
|
||||||
tar -C /tmp/helm/ -zxvf /tmp/helm.tar.gz
|
|
||||||
- name: "Add helm binary to /usr/local/bin"
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
sudo cp /tmp/helm/linux-amd64/helm /usr/local/bin
|
|
||||||
- name: "Cleanup tmp"
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
rm -rf /tmp/helm/ /tmp/helm.tar.gz
|
|
||||||
- name: "Verify correct installation of binary"
|
|
||||||
shell: bash
|
|
||||||
run: |
|
|
||||||
helm version
|
|
||||||
38
.github/actions/kubeconform/action.yml
vendored
Normal file
38
.github/actions/kubeconform/action.yml
vendored
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
name: Setup kubeconform
|
||||||
|
description: A GitHub Action for running kubeconform commands
|
||||||
|
author: Stefan Prodan
|
||||||
|
branding:
|
||||||
|
color: blue
|
||||||
|
icon: command
|
||||||
|
inputs:
|
||||||
|
version:
|
||||||
|
description: "kubeconform version e.g. 0.5.0 (defaults to latest stable release)"
|
||||||
|
required: false
|
||||||
|
arch:
|
||||||
|
description: "arch can be amd64 or arm64"
|
||||||
|
required: true
|
||||||
|
default: "amd64"
|
||||||
|
runs:
|
||||||
|
using: composite
|
||||||
|
steps:
|
||||||
|
- name: "Download binary to the GH runner cache"
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
ARCH=${{ inputs.arch }}
|
||||||
|
VERSION=${{ inputs.version }}
|
||||||
|
|
||||||
|
if [ -z $VERSION ]; then
|
||||||
|
VERSION=$(curl https://api.github.com/repos/yannh/kubeconform/releases/latest -sL | grep tag_name | sed -E 's/.*"([^"]+)".*/\1/' | cut -c 2-)
|
||||||
|
fi
|
||||||
|
|
||||||
|
BIN_URL="https://github.com/yannh/kubeconform/releases/download/v${VERSION}/kubeconform-linux-${ARCH}.tar.gz"
|
||||||
|
BIN_DIR=$RUNNER_TOOL_CACHE/kubeconform/$VERSION/$ARCH
|
||||||
|
|
||||||
|
if [[ ! -x "$BIN_DIR/kind" ]]; then
|
||||||
|
mkdir -p $BIN_DIR
|
||||||
|
cd $BIN_DIR
|
||||||
|
curl -sL $BIN_URL | tar xz
|
||||||
|
chmod +x kubeconform
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$BIN_DIR" >> "$GITHUB_PATH"
|
||||||
6
.github/actions/release-notes/Dockerfile
vendored
6
.github/actions/release-notes/Dockerfile
vendored
@@ -1,6 +0,0 @@
|
|||||||
FROM stefanprodan/alpine-base:latest
|
|
||||||
|
|
||||||
COPY entrypoint.sh /entrypoint.sh
|
|
||||||
RUN chmod +x /entrypoint.sh
|
|
||||||
|
|
||||||
ENTRYPOINT ["/entrypoint.sh"]
|
|
||||||
9
.github/actions/release-notes/action.yml
vendored
9
.github/actions/release-notes/action.yml
vendored
@@ -1,9 +0,0 @@
|
|||||||
name: 'github-release-notes'
|
|
||||||
description: 'A GitHub Action to run github-release-notes commands'
|
|
||||||
author: 'Stefan Prodan'
|
|
||||||
branding:
|
|
||||||
icon: 'command'
|
|
||||||
color: 'blue'
|
|
||||||
runs:
|
|
||||||
using: 'docker'
|
|
||||||
image: 'Dockerfile'
|
|
||||||
25
.github/actions/release-notes/entrypoint.sh
vendored
25
.github/actions/release-notes/entrypoint.sh
vendored
@@ -1,25 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
set -o errexit
|
|
||||||
set -o pipefail
|
|
||||||
|
|
||||||
VERSION=0.2.0
|
|
||||||
BIN_DIR="$GITHUB_WORKSPACE/bin"
|
|
||||||
|
|
||||||
main() {
|
|
||||||
mkdir -p ${BIN_DIR}
|
|
||||||
tmpDir=$(mktemp -d)
|
|
||||||
|
|
||||||
pushd $tmpDir >& /dev/null
|
|
||||||
|
|
||||||
curl -sSL https://github.com/buchanae/github-release-notes/releases/download/${VERSION}/github-release-notes-linux-amd64-${VERSION}.tar.gz | tar xz
|
|
||||||
cp github-release-notes ${BIN_DIR}/github-release-notes
|
|
||||||
|
|
||||||
popd >& /dev/null
|
|
||||||
rm -rf $tmpDir
|
|
||||||
}
|
|
||||||
|
|
||||||
main
|
|
||||||
|
|
||||||
echo "$BIN_DIR" >> $GITHUB_PATH
|
|
||||||
echo "$RUNNER_WORKSPACE/$(basename $GITHUB_REPOSITORY)/bin" >> $GITHUB_PATH
|
|
||||||
11
.github/dependabot.yaml
vendored
Normal file
11
.github/dependabot.yaml
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
version: 2
|
||||||
|
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
groups:
|
||||||
|
actions:
|
||||||
|
patterns:
|
||||||
|
- "*"
|
||||||
|
schedule:
|
||||||
|
interval: "weekly"
|
||||||
51
.github/policy/kubernetes.rego
vendored
51
.github/policy/kubernetes.rego
vendored
@@ -1,51 +0,0 @@
|
|||||||
package kubernetes
|
|
||||||
|
|
||||||
name = input.metadata.name
|
|
||||||
|
|
||||||
kind = input.kind
|
|
||||||
|
|
||||||
is_service {
|
|
||||||
input.kind = "Service"
|
|
||||||
}
|
|
||||||
|
|
||||||
is_deployment {
|
|
||||||
input.kind = "Deployment"
|
|
||||||
}
|
|
||||||
|
|
||||||
is_pod {
|
|
||||||
input.kind = "Pod"
|
|
||||||
}
|
|
||||||
|
|
||||||
split_image(image) = [image, "latest"] {
|
|
||||||
not contains(image, ":")
|
|
||||||
}
|
|
||||||
|
|
||||||
split_image(image) = [image_name, tag] {
|
|
||||||
[image_name, tag] = split(image, ":")
|
|
||||||
}
|
|
||||||
|
|
||||||
pod_containers(pod) = all_containers {
|
|
||||||
keys = {"containers", "initContainers"}
|
|
||||||
all_containers = [c | keys[k]; c = pod.spec[k][_]]
|
|
||||||
}
|
|
||||||
|
|
||||||
containers[container] {
|
|
||||||
pods[pod]
|
|
||||||
all_containers = pod_containers(pod)
|
|
||||||
container = all_containers[_]
|
|
||||||
}
|
|
||||||
|
|
||||||
containers[container] {
|
|
||||||
all_containers = pod_containers(input)
|
|
||||||
container = all_containers[_]
|
|
||||||
}
|
|
||||||
|
|
||||||
pods[pod] {
|
|
||||||
is_deployment
|
|
||||||
pod = input.spec.template
|
|
||||||
}
|
|
||||||
|
|
||||||
pods[pod] {
|
|
||||||
is_pod
|
|
||||||
pod = input
|
|
||||||
}
|
|
||||||
43
.github/policy/rules.rego
vendored
43
.github/policy/rules.rego
vendored
@@ -1,43 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import data.kubernetes
|
|
||||||
|
|
||||||
name = input.metadata.name
|
|
||||||
|
|
||||||
# Deny containers with latest image tag
|
|
||||||
deny[msg] {
|
|
||||||
kubernetes.containers[container]
|
|
||||||
[image_name, "latest"] = kubernetes.split_image(container.image)
|
|
||||||
msg = sprintf("%s in the %s %s has an image %s, using the latest tag", [container.name, kubernetes.kind, kubernetes.name, image_name])
|
|
||||||
}
|
|
||||||
|
|
||||||
# Deny services without app label selector
|
|
||||||
service_labels {
|
|
||||||
input.spec.selector["app"]
|
|
||||||
}
|
|
||||||
deny[msg] {
|
|
||||||
kubernetes.is_service
|
|
||||||
not service_labels
|
|
||||||
msg = sprintf("Service %s should set app label selector", [name])
|
|
||||||
}
|
|
||||||
|
|
||||||
# Deny deployments without app label selector
|
|
||||||
match_labels {
|
|
||||||
input.spec.selector.matchLabels["app"]
|
|
||||||
}
|
|
||||||
deny[msg] {
|
|
||||||
kubernetes.is_deployment
|
|
||||||
not match_labels
|
|
||||||
msg = sprintf("Service %s should set app label selector", [name])
|
|
||||||
}
|
|
||||||
|
|
||||||
# Warn if deployments have no prometheus pod annotations
|
|
||||||
annotations {
|
|
||||||
input.spec.template.metadata.annotations["prometheus.io/scrape"]
|
|
||||||
input.spec.template.metadata.annotations["prometheus.io/port"]
|
|
||||||
}
|
|
||||||
warn[msg] {
|
|
||||||
kubernetes.is_deployment
|
|
||||||
not annotations
|
|
||||||
msg = sprintf("Deployment %s should set prometheus.io/scrape and prometheus.io/port pod annotations", [name])
|
|
||||||
}
|
|
||||||
31
.github/workflows/cve-scan.yml
vendored
31
.github/workflows/cve-scan.yml
vendored
@@ -1,28 +1,23 @@
|
|||||||
name: cve-scan
|
name: cve-scan
|
||||||
|
|
||||||
on:
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- 'master'
|
- "master"
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- "master"
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
trivy:
|
govulncheck:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v5
|
||||||
- name: Build image
|
- name: Vulnerability scan
|
||||||
id: build
|
id: govulncheck
|
||||||
run: |
|
uses: golang/govulncheck-action@v1
|
||||||
IMAGE=test/podinfo:${GITHUB_SHA}
|
|
||||||
docker build -t ${IMAGE} .
|
|
||||||
echo "::set-output name=image::$IMAGE"
|
|
||||||
- name: Run Trivy vulnerability scanner
|
|
||||||
uses: aquasecurity/trivy-action@master
|
|
||||||
with:
|
|
||||||
image-ref: ${{ steps.build.outputs.image }}
|
|
||||||
format: table
|
|
||||||
exit-code: "1"
|
|
||||||
ignore-unfixed: true
|
|
||||||
vuln-type: os,library
|
|
||||||
severity: CRITICAL,HIGH
|
|
||||||
|
|||||||
58
.github/workflows/e2e.yml
vendored
58
.github/workflows/e2e.yml
vendored
@@ -6,28 +6,27 @@ on:
|
|||||||
branches:
|
branches:
|
||||||
- 'master'
|
- 'master'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
kind-helm:
|
kind-helm:
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
helm-version:
|
|
||||||
- 3.8.1
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v5
|
||||||
- name: Setup Kubernetes
|
- name: Setup Kubernetes
|
||||||
uses: engineerd/setup-kind@v0.5.0
|
uses: helm/kind-action@v1.12.0
|
||||||
with:
|
with:
|
||||||
version: v0.11.1
|
cluster_name: kind
|
||||||
- name: Build container image
|
- name: Build container image
|
||||||
run: |
|
run: |
|
||||||
./test/build.sh
|
./test/build.sh
|
||||||
kind load docker-image test/podinfo:latest
|
kind load docker-image test/podinfo:latest
|
||||||
- name: Setup Helm
|
- name: Setup Helm
|
||||||
uses: ./.github/actions/helm
|
uses: azure/setup-helm@v4
|
||||||
with:
|
with:
|
||||||
version: ${{ matrix.helm-version }}
|
version: v3.17.3
|
||||||
- name: Deploy
|
- name: Deploy
|
||||||
run: ./test/deploy.sh
|
run: ./test/deploy.sh
|
||||||
- name: Run integration tests
|
- name: Run integration tests
|
||||||
@@ -36,3 +35,44 @@ jobs:
|
|||||||
if: failure()
|
if: failure()
|
||||||
run: |
|
run: |
|
||||||
kubectl logs -l app=podinfo || true
|
kubectl logs -l app=podinfo || true
|
||||||
|
kind-timoni:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
services:
|
||||||
|
registry:
|
||||||
|
image: registry:2
|
||||||
|
ports:
|
||||||
|
- 5000:5000
|
||||||
|
env:
|
||||||
|
PODINFO_IMAGE_URL: "test/podinfo"
|
||||||
|
PODINFO_MODULE_URL: "oci://localhost:5000/podinfo"
|
||||||
|
PODINFO_VERSION: "0.0.0-devel"
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
- name: Setup Timoni
|
||||||
|
uses: stefanprodan/timoni/actions/setup@main
|
||||||
|
- name: Setup Kubernetes
|
||||||
|
uses: helm/kind-action@v1.12.0
|
||||||
|
with:
|
||||||
|
cluster_name: kind
|
||||||
|
- name: Build container
|
||||||
|
run: |
|
||||||
|
docker build -t ${PODINFO_IMAGE_URL}:${PODINFO_VERSION} --build-arg "REVISION=${GITHUB_SHA}" -f Dockerfile.xx .
|
||||||
|
kind load docker-image ${PODINFO_IMAGE_URL}:${PODINFO_VERSION}
|
||||||
|
- name: Vet module
|
||||||
|
run: |
|
||||||
|
timoni mod vet ./timoni/podinfo --debug
|
||||||
|
- name: Build module
|
||||||
|
run: |
|
||||||
|
timoni mod push ./timoni/podinfo ${PODINFO_MODULE_URL} -v ${PODINFO_VERSION}
|
||||||
|
- name: Apply bundle
|
||||||
|
run: |
|
||||||
|
timoni bundle apply -f ./timoni/bundles/test.podinfo.cue --runtime-from-env
|
||||||
|
- name: Verify status
|
||||||
|
run: |
|
||||||
|
timoni -n podinfo status backend
|
||||||
|
timoni -n podinfo status frontend
|
||||||
|
- name: Debug failure
|
||||||
|
if: failure()
|
||||||
|
run: |
|
||||||
|
kubectl -n podinfo get all || true
|
||||||
|
|||||||
144
.github/workflows/release.yml
vendored
144
.github/workflows/release.yml
vendored
@@ -6,36 +6,55 @@ on:
|
|||||||
- '*'
|
- '*'
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: write # needed to write releases
|
contents: read
|
||||||
id-token: write # needed for keyless signing
|
|
||||||
packages: write # needed for ghcr access
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: write # needed to write releases
|
||||||
|
id-token: write # needed for keyless signing
|
||||||
|
packages: write # needed for ghcr access
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v5
|
||||||
- uses: imjasonh/setup-crane@v0.1
|
- uses: sigstore/cosign-installer@v3
|
||||||
- uses: sigstore/cosign-installer@main
|
- uses: fluxcd/flux2/action@v2.6.4
|
||||||
- name: Setup Helm
|
- uses: stefanprodan/timoni/actions/setup@v0.25.2
|
||||||
uses: ./.github/actions/helm
|
- name: Setup Notation CLI
|
||||||
|
uses: notaryproject/notation-action/setup@v1
|
||||||
with:
|
with:
|
||||||
version: 3.8.1
|
version: "1.1.0"
|
||||||
|
- name: Setup Notation signing keys
|
||||||
|
run: |
|
||||||
|
mkdir -p ~/.config/notation/localkeys/
|
||||||
|
cp ./.notation/signingkeys.json ~/.config/notation/
|
||||||
|
cp ./.notation/notation.crt ~/.config/notation/localkeys/
|
||||||
|
echo "$NOTATION_KEY" > ~/.config/notation/localkeys/notation.key
|
||||||
|
env:
|
||||||
|
NOTATION_KEY: ${{ secrets.NOTATION_SIGNING_KEY }}
|
||||||
|
- name: Setup Go
|
||||||
|
uses: actions/setup-go@v6
|
||||||
|
with:
|
||||||
|
go-version: 1.25.x
|
||||||
|
- name: Setup Helm
|
||||||
|
uses: azure/setup-helm@v4
|
||||||
|
with:
|
||||||
|
version: v3.17.3
|
||||||
- name: Setup QEMU
|
- name: Setup QEMU
|
||||||
uses: docker/setup-qemu-action@v1
|
uses: docker/setup-qemu-action@v3
|
||||||
with:
|
with:
|
||||||
platforms: all
|
platforms: all
|
||||||
- name: Setup Docker Buildx
|
- name: Setup Docker Buildx
|
||||||
id: buildx
|
id: buildx
|
||||||
uses: docker/setup-buildx-action@v1
|
uses: docker/setup-buildx-action@v3
|
||||||
- name: Login to GitHub Container Registry
|
- name: Login to GitHub Container Registry
|
||||||
uses: docker/login-action@v1
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ secrets.DOCKER_USERNAME }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ secrets.GHCR_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
- name: Login to Docker Hub
|
- name: Login to Docker Hub
|
||||||
uses: docker/login-action@v1
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKER_USERNAME }}
|
username: ${{ secrets.DOCKER_USERNAME }}
|
||||||
password: ${{ secrets.DOCKER_PASSWORD }}
|
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
@@ -46,43 +65,64 @@ jobs:
|
|||||||
if [[ $GITHUB_REF == refs/tags/* ]]; then
|
if [[ $GITHUB_REF == refs/tags/* ]]; then
|
||||||
VERSION=${GITHUB_REF/refs\/tags\//}
|
VERSION=${GITHUB_REF/refs\/tags\//}
|
||||||
fi
|
fi
|
||||||
echo ::set-output name=BUILD_DATE::$(date -u +'%Y-%m-%dT%H:%M:%SZ')
|
echo "BUILD_DATE=$(date -u +'%Y-%m-%dT%H:%M:%SZ')" >> $GITHUB_OUTPUT
|
||||||
echo ::set-output name=VERSION::${VERSION}
|
echo "VERSION=${VERSION}" >> $GITHUB_OUTPUT
|
||||||
- name: Publish multi-arch image
|
echo "REVISION=${GITHUB_SHA}" >> $GITHUB_OUTPUT
|
||||||
uses: docker/build-push-action@v2
|
- name: Generate images meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
with:
|
with:
|
||||||
|
images: |
|
||||||
|
docker.io/stefanprodan/podinfo
|
||||||
|
ghcr.io/stefanprodan/podinfo
|
||||||
|
tags: |
|
||||||
|
type=raw,value=${{ steps.prep.outputs.VERSION }}
|
||||||
|
type=raw,value=latest
|
||||||
|
- name: Publish multi-arch image
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
sbom: true
|
||||||
|
provenance: true
|
||||||
push: true
|
push: true
|
||||||
builder: ${{ steps.buildx.outputs.name }}
|
builder: ${{ steps.buildx.outputs.name }}
|
||||||
context: .
|
context: .
|
||||||
file: ./Dockerfile.xx
|
file: ./Dockerfile.xx
|
||||||
|
build-args: |
|
||||||
|
REVISION=${{ steps.prep.outputs.REVISION }}
|
||||||
platforms: linux/amd64,linux/arm/v7,linux/arm64
|
platforms: linux/amd64,linux/arm/v7,linux/arm64
|
||||||
tags: |
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
docker.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
docker.io/stefanprodan/podinfo:latest
|
- name: Publish Timoni module to GHCR
|
||||||
ghcr.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }}
|
run: |
|
||||||
labels: |
|
timoni mod push ./timoni/podinfo oci://ghcr.io/stefanprodan/modules/podinfo \
|
||||||
org.opencontainers.image.title=${{ github.event.repository.name }}
|
--sign cosign \
|
||||||
org.opencontainers.image.description=${{ github.event.repository.description }}
|
--version ${{ steps.prep.outputs.VERSION }} \
|
||||||
org.opencontainers.image.source=${{ github.event.repository.html_url }}
|
-a 'org.opencontainers.image.source=https://github.com/stefanprodan/podinfo' \
|
||||||
org.opencontainers.image.url=${{ github.event.repository.html_url }}
|
-a 'org.opencontainers.image.licenses=Apache-2.0' \
|
||||||
org.opencontainers.image.revision=${{ github.sha }}
|
-a 'org.opencontainers.image.description=A timoni.sh module for deploying Podinfo.' \
|
||||||
org.opencontainers.image.version=${{ steps.prep.outputs.VERSION }}
|
-a 'org.opencontainers.image.documentation=https://github.com/stefanprodan/podinfo/blob/main/timoni/podinfo/README.md'
|
||||||
org.opencontainers.image.created=${{ steps.prep.outputs.BUILD_DATE }}
|
|
||||||
- name: Publish Helm chart to GHCR
|
- name: Publish Helm chart to GHCR
|
||||||
run: |
|
run: |
|
||||||
helm package charts/podinfo
|
helm package charts/podinfo
|
||||||
helm push podinfo-${{ steps.prep.outputs.VERSION }}.tgz oci://ghcr.io/stefanprodan/charts
|
helm push podinfo-${{ steps.prep.outputs.VERSION }}.tgz oci://ghcr.io/stefanprodan/charts
|
||||||
rm podinfo-${{ steps.prep.outputs.VERSION }}.tgz
|
rm podinfo-${{ steps.prep.outputs.VERSION }}.tgz
|
||||||
- name: Sign images
|
- name: Publish Flux OCI artifact to GHCR
|
||||||
|
run: |
|
||||||
|
flux push artifact oci://ghcr.io/stefanprodan/manifests/podinfo:${{ steps.prep.outputs.VERSION }} \
|
||||||
|
--path="./kustomize" \
|
||||||
|
--source="${{ github.event.repository.html_url }}" \
|
||||||
|
--revision="${GITHUB_REF_NAME}/${GITHUB_SHA}"
|
||||||
|
flux tag artifact oci://ghcr.io/stefanprodan/manifests/podinfo:${{ steps.prep.outputs.VERSION }} --tag latest
|
||||||
|
- name: Sign artifacts with Cosign
|
||||||
env:
|
env:
|
||||||
COSIGN_EXPERIMENTAL: 1
|
COSIGN_EXPERIMENTAL: 1
|
||||||
run: |
|
run: |
|
||||||
cosign sign docker.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }}
|
cosign sign docker.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }} --yes
|
||||||
cosign sign docker.io/stefanprodan/podinfo:latest
|
cosign sign ghcr.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }} --yes
|
||||||
cosign sign ghcr.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }}
|
cosign sign ghcr.io/stefanprodan/charts/podinfo:${{ steps.prep.outputs.VERSION }} --yes
|
||||||
cosign sign ghcr.io/stefanprodan/charts/podinfo:${{ steps.prep.outputs.VERSION }}
|
cosign sign ghcr.io/stefanprodan/manifests/podinfo:${{ steps.prep.outputs.VERSION }} --yes
|
||||||
- name: Publish base image
|
- name: Publish base image
|
||||||
uses: docker/build-push-action@v2
|
uses: docker/build-push-action@v6
|
||||||
with:
|
with:
|
||||||
push: true
|
push: true
|
||||||
builder: ${{ steps.buildx.outputs.name }}
|
builder: ${{ steps.buildx.outputs.name }}
|
||||||
@@ -96,28 +136,30 @@ jobs:
|
|||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
- name: Publish config artifact
|
- name: Publish config artifact
|
||||||
run: |
|
run: |
|
||||||
cd kustomize
|
flux push artifact oci://ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }} \
|
||||||
tar -cf config.tar * --numeric-owner --owner=0 --group=0
|
--path="./kustomize" \
|
||||||
crane append -f config.tar -t ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }}
|
--source="${{ github.event.repository.html_url }}" \
|
||||||
crane tag ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }} latest
|
--revision="${GITHUB_REF_NAME}/${GITHUB_SHA}"
|
||||||
rm config.tar
|
flux tag artifact oci://ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }} --tag latest
|
||||||
- name: Sign config artifact
|
- name: Sign config artifact with cso
|
||||||
run: |
|
run: |
|
||||||
echo "$COSIGN_KEY" > /tmp/cosign.key
|
echo "$COSIGN_KEY" > /tmp/cosign.key
|
||||||
cosign sign -key /tmp/cosign.key ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }}
|
cosign sign -key /tmp/cosign.key ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }} --yes
|
||||||
cosign sign -key /tmp/cosign.key ghcr.io/stefanprodan/podinfo-deploy:latest
|
cosign sign -key /tmp/cosign.key ghcr.io/stefanprodan/podinfo-deploy:latest --yes
|
||||||
env:
|
env:
|
||||||
COSIGN_PASSWORD: ${{secrets.COSIGN_PASSWORD}}
|
COSIGN_PASSWORD: ${{secrets.COSIGN_PASSWORD}}
|
||||||
COSIGN_KEY: ${{secrets.COSIGN_KEY}}
|
COSIGN_KEY: ${{secrets.COSIGN_KEY}}
|
||||||
- uses: ./.github/actions/release-notes
|
- name: Sign artifacts with Notation
|
||||||
- name: Generate release notes
|
|
||||||
run: |
|
run: |
|
||||||
echo 'CHANGELOG' > /tmp/release.txt
|
notation sign --signature-format cose ghcr.io/stefanprodan/podinfo:${{ steps.prep.outputs.VERSION }}
|
||||||
github-release-notes -org stefanprodan -repo podinfo -since-latest-release >> /tmp/release.txt
|
notation sign --signature-format cose ghcr.io/stefanprodan/charts/podinfo:${{ steps.prep.outputs.VERSION }}
|
||||||
|
notation sign --signature-format cose ghcr.io/stefanprodan/manifests/podinfo:${{ steps.prep.outputs.VERSION }}
|
||||||
|
notation sign --signature-format cose ghcr.io/stefanprodan/podinfo-deploy:${{ steps.prep.outputs.VERSION }}
|
||||||
|
notation sign --signature-format cose ghcr.io/stefanprodan/podinfo-deploy:latest
|
||||||
- name: Publish release
|
- name: Publish release
|
||||||
uses: goreleaser/goreleaser-action@v1
|
uses: goreleaser/goreleaser-action@v6
|
||||||
with:
|
with:
|
||||||
version: latest
|
version: latest
|
||||||
args: release --release-notes=/tmp/release.txt --skip-validate
|
args: release --skip=validate
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|||||||
74
.github/workflows/test.yml
vendored
74
.github/workflows/test.yml
vendored
@@ -6,42 +6,68 @@ on:
|
|||||||
branches:
|
branches:
|
||||||
- 'master'
|
- 'master'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
env:
|
||||||
|
KUBERNETES_VERSION: 1.31.0
|
||||||
|
HELM_VERSION: 3.17.3
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v5
|
||||||
- name: Restore Go cache
|
|
||||||
uses: actions/cache@v1
|
|
||||||
with:
|
|
||||||
path: ~/go/pkg/mod
|
|
||||||
key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
|
|
||||||
restore-keys: ${{ runner.os }}-go-
|
|
||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
uses: actions/setup-go@v2
|
uses: actions/setup-go@v6
|
||||||
with:
|
with:
|
||||||
go-version: 1.17.x
|
go-version: 1.25.x
|
||||||
|
cache-dependency-path: |
|
||||||
|
**/go.sum
|
||||||
|
**/go.mod
|
||||||
|
- name: Setup kubectl
|
||||||
|
uses: azure/setup-kubectl@v4
|
||||||
|
with:
|
||||||
|
version: v${{ env.KUBERNETES_VERSION }}
|
||||||
|
- name: Setup kubeconform
|
||||||
|
uses: ./.github/actions/kubeconform
|
||||||
|
- name: Setup Helm
|
||||||
|
uses: azure/setup-helm@v4
|
||||||
|
with:
|
||||||
|
version: v${{ env.HELM_VERSION }}
|
||||||
|
- name: Setup CUE
|
||||||
|
uses: cue-lang/setup-cue@v1.0.1
|
||||||
|
- name: Setup Timoni
|
||||||
|
uses: stefanprodan/timoni/actions/setup@v0.25.2
|
||||||
- name: Run unit tests
|
- name: Run unit tests
|
||||||
run: make test
|
run: make test
|
||||||
|
- name: Validate Helm chart
|
||||||
|
run: |
|
||||||
|
helm lint ./charts/podinfo/
|
||||||
|
helm template ./charts/podinfo/ | kubeconform -strict -summary -kubernetes-version ${{ env.KUBERNETES_VERSION }}
|
||||||
|
- name: Validate Kustomize overlay
|
||||||
|
run: |
|
||||||
|
kubectl kustomize ./kustomize/ | kubeconform -strict -summary -kubernetes-version ${{ env.KUBERNETES_VERSION }}
|
||||||
|
- name: Verify CUE formatting
|
||||||
|
working-directory: ./timoni/podinfo
|
||||||
|
run: |
|
||||||
|
cue fmt ./...
|
||||||
|
status=$(git status . --porcelain)
|
||||||
|
[[ -z "$status" ]] || {
|
||||||
|
echo "CUE files are not correctly formatted"
|
||||||
|
echo "$status"
|
||||||
|
git diff
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
- name: Validate Timoni module
|
||||||
|
working-directory: ./timoni/podinfo
|
||||||
|
run: |
|
||||||
|
timoni mod lint .
|
||||||
|
timoni build podinfo . -f test_values.cue | kubeconform -strict -summary -skip=ServiceMonitor -kubernetes-version ${{ env.KUBERNETES_VERSION }}
|
||||||
- name: Check if working tree is dirty
|
- name: Check if working tree is dirty
|
||||||
run: |
|
run: |
|
||||||
if [[ $(git diff --stat) != '' ]]; then
|
if [[ $(git diff --stat) != '' ]]; then
|
||||||
echo 'run make test and commit changes'
|
echo 'run make test and commit changes'
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
- name: Validate Helm chart
|
|
||||||
uses: stefanprodan/kube-tools@v1
|
|
||||||
with:
|
|
||||||
kubectl: 1.19.11
|
|
||||||
helm: 2.17.0
|
|
||||||
helmv3: 3.6.0
|
|
||||||
command: |
|
|
||||||
helmv3 template ./charts/podinfo | kubeval --strict --kubernetes-version 1.19.11 --schema-location https://raw.githubusercontent.com/yannh/kubernetes-json-schema/master
|
|
||||||
- name: Validate kustomization
|
|
||||||
uses: stefanprodan/kube-tools@v1
|
|
||||||
with:
|
|
||||||
kubectl: 1.19.11
|
|
||||||
command: |
|
|
||||||
kustomize build ./kustomize | kubeval --strict --kubernetes-version 1.19.11 --schema-location https://raw.githubusercontent.com/yannh/kubernetes-json-schema/master
|
|
||||||
kustomize build ./kustomize | conftest test -p .github/policy -
|
|
||||||
|
|||||||
8
.gitignore
vendored
8
.gitignore
vendored
@@ -19,4 +19,10 @@ release/
|
|||||||
build/
|
build/
|
||||||
gcloud/
|
gcloud/
|
||||||
dist/
|
dist/
|
||||||
bin/
|
bin/
|
||||||
|
cue/cue.mod/gen/
|
||||||
|
cue/go.mod
|
||||||
|
cue/go.sum
|
||||||
|
|
||||||
|
.notation/podinfo.csr
|
||||||
|
.notation/podinfo.key
|
||||||
|
|||||||
@@ -1,3 +1,18 @@
|
|||||||
|
version: 2
|
||||||
|
|
||||||
|
# xref: https://goreleaser.com/customization/project/
|
||||||
|
project_name: podinfo
|
||||||
|
|
||||||
|
# xref: https://goreleaser.com/customization/hooks/
|
||||||
|
before:
|
||||||
|
hooks:
|
||||||
|
- go mod download
|
||||||
|
|
||||||
|
# xref: https://goreleaser.com/customization/env/
|
||||||
|
env:
|
||||||
|
- CGO_ENABLED=0
|
||||||
|
|
||||||
|
# xref: https://goreleaser.com/customization/build/
|
||||||
builds:
|
builds:
|
||||||
- main: ./cmd/podcli
|
- main: ./cmd/podcli
|
||||||
binary: podcli
|
binary: podcli
|
||||||
@@ -8,9 +23,13 @@ builds:
|
|||||||
- linux
|
- linux
|
||||||
goarch:
|
goarch:
|
||||||
- amd64
|
- amd64
|
||||||
env:
|
|
||||||
- CGO_ENABLED=0
|
# xref: https://goreleaser.com/customization/archive/
|
||||||
archives:
|
archives:
|
||||||
- name_template: "{{ .Binary }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}"
|
- name_template: "{{ .Binary }}_{{ .Version }}_{{ .Os }}_{{ .Arch }}"
|
||||||
files:
|
files:
|
||||||
- none*
|
- LICENSE
|
||||||
|
|
||||||
|
# xref: https://goreleaser.com/customization/changelog/
|
||||||
|
changelog:
|
||||||
|
use: github-native
|
||||||
|
|||||||
15
.notation/README.md
Normal file
15
.notation/README.md
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Podinfo signed releases
|
||||||
|
|
||||||
|
Podinfo release assets such as the Helm chart and the Flux artifact
|
||||||
|
are published to GitHub Container Registry and are signed with
|
||||||
|
[Notation](https://github.com/notaryproject/notation).
|
||||||
|
|
||||||
|
## Generate signing keys
|
||||||
|
|
||||||
|
Generate a new signing key pair:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
openssl genrsa -out podinfo.key 2048
|
||||||
|
openssl req -new -key podinfo.key -out podinfo.csr -config codesign.cnf
|
||||||
|
openssl x509 -req -days 1826 -in podinfo.csr -signkey podinfo.key -out notation.crt -extensions v3_req -extfile codesign.cnf
|
||||||
|
```
|
||||||
18
.notation/codesign.cnf
Normal file
18
.notation/codesign.cnf
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
[ req ]
|
||||||
|
default_bits = 2048
|
||||||
|
default_keyfile = privatekey.pem
|
||||||
|
distinguished_name = req_distinguished_name
|
||||||
|
req_extensions = v3_req
|
||||||
|
prompt = no
|
||||||
|
|
||||||
|
[ req_distinguished_name ]
|
||||||
|
C = RO
|
||||||
|
ST = BU
|
||||||
|
L = Bucharest
|
||||||
|
O = Notary
|
||||||
|
CN = stefanprodan.com
|
||||||
|
|
||||||
|
[ v3_req ]
|
||||||
|
keyUsage = critical,digitalSignature
|
||||||
|
extendedKeyUsage = critical,codeSigning
|
||||||
|
#subjectKeyIdentifier = hash
|
||||||
21
.notation/notation.crt
Normal file
21
.notation/notation.crt
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
MIIDbDCCAlSgAwIBAgIUP7zhmTw5XTWLcgBGkBEsErMOkz4wDQYJKoZIhvcNAQEL
|
||||||
|
BQAwWjELMAkGA1UEBhMCUk8xCzAJBgNVBAgMAkJVMRIwEAYDVQQHDAlCdWNoYXJl
|
||||||
|
c3QxDzANBgNVBAoMBk5vdGFyeTEZMBcGA1UEAwwQc3RlZmFucHJvZGFuLmNvbTAe
|
||||||
|
Fw0yNDAyMjUxMDAyMzZaFw0yOTAyMjQxMDAyMzZaMFoxCzAJBgNVBAYTAlJPMQsw
|
||||||
|
CQYDVQQIDAJCVTESMBAGA1UEBwwJQnVjaGFyZXN0MQ8wDQYDVQQKDAZOb3Rhcnkx
|
||||||
|
GTAXBgNVBAMMEHN0ZWZhbnByb2Rhbi5jb20wggEiMA0GCSqGSIb3DQEBAQUAA4IB
|
||||||
|
DwAwggEKAoIBAQDtH4oPi3SyX/DGv6NdjIvmApvD9eeSgsmHdwpAly8T9D2me+fx
|
||||||
|
Z+wRNJmq4aq/A1anX+Sg28iwHzV+1WKpsHnjYzDAJSEYP2S8A5H1nGRKUoibdijw
|
||||||
|
C3QBh5C75rjF/tmZVSX/Vgbf3HJJEsF4WUxWabLxoV2QLo7UlEsQd9+bSeKNMncx
|
||||||
|
1+E6FdbRCrYo90iobvZJ8K/S2zCWq/JTeHfTnmSEDhx6nMJcaSjvMPn3zyauWcQw
|
||||||
|
dDpkcaGiJ64fEJRT2OFxXv9u+vDmIMKzo/Wjbd+IzFj6YY4VisK88aU7tmDelnk5
|
||||||
|
gQB9eu62PFoaVsYJp4VOhblFKvGJpQwbWB9BAgMBAAGjKjAoMA4GA1UdDwEB/wQE
|
||||||
|
AwIHgDAWBgNVHSUBAf8EDDAKBggrBgEFBQcDAzANBgkqhkiG9w0BAQsFAAOCAQEA
|
||||||
|
6x+C6hAIbLwMvkNx4K5p7Qe/pLQR0VwQFAw10yr/5KSN+YKFpon6pQ0TebL7qll+
|
||||||
|
uBGZvtQhN6v+DlnVqB7lvJKd+89isgirkkews5KwuXg7Gv5UPIugH0dXISZU8DMJ
|
||||||
|
7J4oKREv5HzdFmfsUfNlQcfyVTjKL6UINXfKGdqNNxXxR9b4a1TY2JcmEhzBTHaq
|
||||||
|
ZqX6HK784a0dB7aHgeFrFwPCCP4M684Hs7CFbk3jo2Ef4ljnB5AyWpe8pwCLMdRt
|
||||||
|
UjSjL5xJWVQvRU+STQsPr6SvpokPCG4rLQyjgeYYk4CCj5piSxbSUZFavq8v1y7Y
|
||||||
|
m91USVqfeUX7ZzjDxPHE2A==
|
||||||
|
-----END CERTIFICATE-----
|
||||||
10
.notation/signingkeys.json
Normal file
10
.notation/signingkeys.json
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
"default": "stefanprodan.com",
|
||||||
|
"keys": [
|
||||||
|
{
|
||||||
|
"name": "stefanprodan.com",
|
||||||
|
"keyPath": "/home/runner/.config/notation/localkeys/notation.key",
|
||||||
|
"certPath": "/home/runner/.config/notation/localkeys/notation.crt"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
19
.notation/trustpolicy.json
Normal file
19
.notation/trustpolicy.json
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"version": "1.0",
|
||||||
|
"trustPolicies": [
|
||||||
|
{
|
||||||
|
"name": "stefanprodan.com",
|
||||||
|
"registryScopes": [
|
||||||
|
"ghcr.io/stefanprodan/podinfo-deploy",
|
||||||
|
"ghcr.io/stefanprodan/charts/podinfo"
|
||||||
|
],
|
||||||
|
"signatureVerification": {
|
||||||
|
"level" : "strict"
|
||||||
|
},
|
||||||
|
"trustStores": [ "ca:stefanprodan.com" ],
|
||||||
|
"trustedIdentities": [
|
||||||
|
"x509.subject: C=RO, ST=BU, L=Bucharest, O=Notary, CN=stefanprodan.com"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
FROM golang:1.17-alpine as builder
|
FROM golang:1.25-alpine AS builder
|
||||||
|
|
||||||
ARG REVISION
|
ARG REVISION
|
||||||
|
|
||||||
@@ -18,7 +18,7 @@ RUN CGO_ENABLED=0 go build -ldflags "-s -w \
|
|||||||
-X github.com/stefanprodan/podinfo/pkg/version.REVISION=${REVISION}" \
|
-X github.com/stefanprodan/podinfo/pkg/version.REVISION=${REVISION}" \
|
||||||
-a -o bin/podcli cmd/podcli/*
|
-a -o bin/podcli cmd/podcli/*
|
||||||
|
|
||||||
FROM alpine:3.15
|
FROM alpine:3.22
|
||||||
|
|
||||||
ARG BUILD_DATE
|
ARG BUILD_DATE
|
||||||
ARG VERSION
|
ARG VERSION
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
FROM golang:1.17
|
FROM golang:1.25
|
||||||
|
|
||||||
WORKDIR /workspace
|
WORKDIR /workspace
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
ARG GO_VERSION=1.17
|
ARG GO_VERSION=1.25
|
||||||
ARG XX_VERSION=1.1.0
|
ARG XX_VERSION=1.6.1
|
||||||
|
|
||||||
FROM --platform=$BUILDPLATFORM tonistiigi/xx:${XX_VERSION} AS xx
|
FROM --platform=$BUILDPLATFORM tonistiigi/xx:${XX_VERSION} AS xx
|
||||||
|
|
||||||
@@ -28,7 +28,7 @@ RUN xx-go build -ldflags "-s -w \
|
|||||||
-X github.com/stefanprodan/podinfo/pkg/version.REVISION=${REVISION}" \
|
-X github.com/stefanprodan/podinfo/pkg/version.REVISION=${REVISION}" \
|
||||||
-a -o bin/podcli cmd/podcli/*
|
-a -o bin/podcli cmd/podcli/*
|
||||||
|
|
||||||
FROM alpine:3.15
|
FROM alpine:3.22
|
||||||
|
|
||||||
ARG BUILD_DATE
|
ARG BUILD_DATE
|
||||||
ARG VERSION
|
ARG VERSION
|
||||||
|
|||||||
25
Makefile
25
Makefile
@@ -16,7 +16,7 @@ run:
|
|||||||
--ui-logo=https://raw.githubusercontent.com/stefanprodan/podinfo/gh-pages/cuddle_clap.gif $(EXTRA_RUN_ARGS)
|
--ui-logo=https://raw.githubusercontent.com/stefanprodan/podinfo/gh-pages/cuddle_clap.gif $(EXTRA_RUN_ARGS)
|
||||||
|
|
||||||
.PHONY: test
|
.PHONY: test
|
||||||
test:
|
test: tidy fmt vet
|
||||||
go test ./... -coverprofile cover.out
|
go test ./... -coverprofile cover.out
|
||||||
|
|
||||||
build:
|
build:
|
||||||
@@ -24,11 +24,13 @@ build:
|
|||||||
GIT_COMMIT=$$(git rev-list -1 HEAD) && CGO_ENABLED=0 go build -ldflags "-s -w -X github.com/stefanprodan/podinfo/pkg/version.REVISION=$(GIT_COMMIT)" -a -o ./bin/podcli ./cmd/podcli/*
|
GIT_COMMIT=$$(git rev-list -1 HEAD) && CGO_ENABLED=0 go build -ldflags "-s -w -X github.com/stefanprodan/podinfo/pkg/version.REVISION=$(GIT_COMMIT)" -a -o ./bin/podcli ./cmd/podcli/*
|
||||||
|
|
||||||
tidy:
|
tidy:
|
||||||
rm -f go.sum; go mod tidy -compat=1.17
|
rm -f go.sum; go mod tidy -compat=1.25
|
||||||
|
|
||||||
|
vet:
|
||||||
|
go vet ./...
|
||||||
|
|
||||||
fmt:
|
fmt:
|
||||||
gofmt -l -s -w ./
|
go fmt ./...
|
||||||
goimports -l -w ./
|
|
||||||
|
|
||||||
build-charts:
|
build-charts:
|
||||||
helm lint charts/*
|
helm lint charts/*
|
||||||
@@ -79,12 +81,19 @@ version-set:
|
|||||||
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/webapp/backend/deployment.yaml && \
|
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/webapp/backend/deployment.yaml && \
|
||||||
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/bases/frontend/deployment.yaml && \
|
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/bases/frontend/deployment.yaml && \
|
||||||
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/bases/backend/deployment.yaml && \
|
/usr/bin/sed -i '' "s/podinfo:$$current/podinfo:$$next/g" deploy/bases/backend/deployment.yaml && \
|
||||||
echo "Version $$next set in code, deployment, chart and kustomize"
|
/usr/bin/sed -i '' "s/$$current/$$next/g" timoni/podinfo/values.cue && \
|
||||||
|
echo "Version $$next set in code, deployment, module, chart and kustomize"
|
||||||
|
|
||||||
release:
|
release:
|
||||||
git tag $(VERSION)
|
git tag -s -m $(VERSION) $(VERSION)
|
||||||
git push origin $(VERSION)
|
git push origin $(VERSION)
|
||||||
|
|
||||||
swagger:
|
swagger:
|
||||||
go get github.com/swaggo/swag/cmd/swag
|
go install github.com/swaggo/swag/cmd/swag@latest
|
||||||
cd pkg/api && $$(go env GOPATH)/bin/swag init -g server.go
|
go get github.com/swaggo/swag/gen@latest
|
||||||
|
go get github.com/swaggo/swag/cmd/swag@latest
|
||||||
|
cd pkg/api/http && $$(go env GOPATH)/bin/swag init -g server.go
|
||||||
|
|
||||||
|
.PHONY: timoni-build
|
||||||
|
timoni-build:
|
||||||
|
@timoni build podinfo ./timoni/podinfo -f ./timoni/podinfo/debug_values.cue
|
||||||
|
|||||||
41
README.md
41
README.md
@@ -20,12 +20,12 @@ Specifications:
|
|||||||
* 12-factor app with viper
|
* 12-factor app with viper
|
||||||
* Fault injection (random errors and latency)
|
* Fault injection (random errors and latency)
|
||||||
* Swagger docs
|
* Swagger docs
|
||||||
* Helm and Kustomize installers
|
* Timoni, Helm and Kustomize installers
|
||||||
* End-to-End testing with Kubernetes Kind and Helm
|
* End-to-End testing with Kubernetes Kind and Helm
|
||||||
* Kustomize testing with GitHub Actions and Open Policy Agent
|
* Multi-arch container image with Docker buildx and GitHub Actions
|
||||||
* Multi-arch container image with Docker buildx and Github Actions
|
|
||||||
* Container image signing with Sigstore cosign
|
* Container image signing with Sigstore cosign
|
||||||
* CVE scanning with Trivy
|
* SBOMs and SLSA Provenance embedded in the container image
|
||||||
|
* CVE scanning with govulncheck
|
||||||
|
|
||||||
Web API:
|
Web API:
|
||||||
|
|
||||||
@@ -57,6 +57,16 @@ Web API:
|
|||||||
gRPC API:
|
gRPC API:
|
||||||
|
|
||||||
* `/grpc.health.v1.Health/Check` health checking
|
* `/grpc.health.v1.Health/Check` health checking
|
||||||
|
* `/grpc.EchoService/Echo` echos the received content
|
||||||
|
* `/grpc.VersionService/Version` returns podinfo version and Git commit hash
|
||||||
|
* `/grpc.DelayService/Delay` returns a successful response after the given seconds in the body of gRPC request
|
||||||
|
* `/grpc.EnvService/Env` returns environment variables as a JSON array
|
||||||
|
* `/grpc.HeaderService/Header` returns the headers present in the gRPC request. Any custom header can also be given as a part of request and that can be returned using this API
|
||||||
|
* `/grpc.InfoService/Info` returns the runtime information
|
||||||
|
* `/grpc.PanicService/Panic` crashes the process with gRPC status code as '1 CANCELLED'
|
||||||
|
* `/grpc.StatusService/Status` returns the gRPC Status code given in the request body
|
||||||
|
* `/grpc.TokenService/TokenGenerate` issues a JWT token valid for one minute
|
||||||
|
* `/grpc.TokenService/TokenValidate` validates the JWT token
|
||||||
|
|
||||||
Web UI:
|
Web UI:
|
||||||
|
|
||||||
@@ -66,16 +76,23 @@ To access the Swagger UI open `<podinfo-host>/swagger/index.html` in a browser.
|
|||||||
|
|
||||||
### Guides
|
### Guides
|
||||||
|
|
||||||
* [GitOps Progressive Deliver with Flagger, Helm v3 and Linkerd](https://helm.workshop.flagger.dev/intro/)
|
* [Getting started with Timoni](https://timoni.sh/quickstart/)
|
||||||
* [GitOps Progressive Deliver on EKS with Flagger and AppMesh](https://eks.handson.flagger.dev/prerequisites/)
|
* [Getting started with Flux](https://fluxcd.io/flux/get-started/)
|
||||||
* [Automated canary deployments with Flagger and Istio](https://medium.com/google-cloud/automated-canary-deployments-with-flagger-and-istio-ac747827f9d1)
|
* [Progressive Deliver with Flagger and Linkerd](https://docs.flagger.app/tutorials/linkerd-progressive-delivery)
|
||||||
* [Kubernetes autoscaling with Istio metrics](https://medium.com/google-cloud/kubernetes-autoscaling-with-istio-metrics-76442253a45a)
|
* [Automated canary deployments with Kubernetes Gateway API](https://docs.flagger.app/tutorials/gatewayapi-progressive-delivery)
|
||||||
* [Autoscaling EKS on Fargate with custom metrics](https://aws.amazon.com/blogs/containers/autoscaling-eks-on-fargate-with-custom-metrics/)
|
|
||||||
* [Managing Helm releases the GitOps way](https://medium.com/google-cloud/managing-helm-releases-the-gitops-way-207a6ac6ff0e)
|
|
||||||
* [Securing EKS Ingress With Contour And Let’s Encrypt The GitOps Way](https://aws.amazon.com/blogs/containers/securing-eks-ingress-contour-lets-encrypt-gitops/)
|
|
||||||
|
|
||||||
### Install
|
### Install
|
||||||
|
|
||||||
|
To install Podinfo on Kubernetes the minimum required version is **Kubernetes v1.23**.
|
||||||
|
|
||||||
|
#### Timoni
|
||||||
|
|
||||||
|
Install with [Timoni](https://timoni.sh):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
timoni -n default apply podinfo oci://ghcr.io/stefanprodan/modules/podinfo
|
||||||
|
```
|
||||||
|
|
||||||
#### Helm
|
#### Helm
|
||||||
|
|
||||||
Install from github.io:
|
Install from github.io:
|
||||||
@@ -89,7 +106,7 @@ helm upgrade --install --wait frontend \
|
|||||||
--set backend=http://backend-podinfo:9898/echo \
|
--set backend=http://backend-podinfo:9898/echo \
|
||||||
podinfo/podinfo
|
podinfo/podinfo
|
||||||
|
|
||||||
helm test frontend
|
helm test frontend --namespace test
|
||||||
|
|
||||||
helm upgrade --install --wait backend \
|
helm upgrade --install --wait backend \
|
||||||
--namespace test \
|
--namespace test \
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
apiVersion: v1
|
apiVersion: v1
|
||||||
version: 6.1.1
|
version: 6.9.2
|
||||||
appVersion: 6.1.1
|
appVersion: 6.9.2
|
||||||
name: podinfo
|
name: podinfo
|
||||||
engine: gotpl
|
engine: gotpl
|
||||||
description: Podinfo Helm chart for Kubernetes
|
description: Podinfo Helm chart for Kubernetes
|
||||||
@@ -10,4 +10,4 @@ maintainers:
|
|||||||
name: stefanprodan
|
name: stefanprodan
|
||||||
sources:
|
sources:
|
||||||
- https://github.com/stefanprodan/podinfo
|
- https://github.com/stefanprodan/podinfo
|
||||||
kubeVersion: ">=1.19.0-0"
|
kubeVersion: ">=1.23.0-0"
|
||||||
|
|||||||
@@ -9,7 +9,23 @@ for end-to-end testing and workshops.
|
|||||||
|
|
||||||
## Installing the Chart
|
## Installing the Chart
|
||||||
|
|
||||||
To install the chart with the release name `my-release`:
|
The Podinfo charts are published to
|
||||||
|
[GitHub Container Registry](https://github.com/stefanprodan/podinfo/pkgs/container/charts%2Fpodinfo)
|
||||||
|
and signed with [Cosign](https://github.com/sigstore/cosign) & GitHub Actions OIDC.
|
||||||
|
|
||||||
|
To install the chart with the release name `my-release` from GHCR:
|
||||||
|
|
||||||
|
```console
|
||||||
|
$ helm upgrade -i my-release oci://ghcr.io/stefanprodan/charts/podinfo
|
||||||
|
```
|
||||||
|
|
||||||
|
To verify a chart with Cosign:
|
||||||
|
|
||||||
|
```console
|
||||||
|
$ cosign verify ghcr.io/stefanprodan/charts/podinfo:<VERSION>
|
||||||
|
```
|
||||||
|
|
||||||
|
Alternatively, you can install the chart from GitHub pages:
|
||||||
|
|
||||||
```console
|
```console
|
||||||
$ helm repo add podinfo https://stefanprodan.github.io/podinfo
|
$ helm repo add podinfo https://stefanprodan.github.io/podinfo
|
||||||
@@ -34,60 +50,67 @@ The command removes all the Kubernetes components associated with the chart and
|
|||||||
|
|
||||||
The following tables lists the configurable parameters of the podinfo chart and their default values.
|
The following tables lists the configurable parameters of the podinfo chart and their default values.
|
||||||
|
|
||||||
Parameter | Default | Description
|
| Parameter | Default | Description |
|
||||||
--- | --- | ---
|
| --------------------------------- | ---------------------- | ---------------------------------------------------------------------------------------------------------------------- |
|
||||||
`replicaCount` | `1` | Desired number of pods
|
| `replicaCount` | `1` | Desired number of pods |
|
||||||
`logLevel` | `info` | Log level: `debug`, `info`, `warn`, `error`
|
| `logLevel` | `info` | Log level: `debug`, `info`, `warn`, `error` |
|
||||||
`backend` | `None` | Echo backend URL
|
| `backend` | `None` | Echo backend URL |
|
||||||
`backends` | `[]` | Array of echo backend URLs
|
| `backends` | `[]` | Array of echo backend URLs |
|
||||||
`cache` | `None` | Redis address in the format `<host>:<port>`
|
| `cache` | `None` | Redis address in the format `tcp://<host>:<port>` |
|
||||||
`redis.enabled` | `false` | Create Redis deployment for caching purposes
|
| `redis.enabled` | `false` | Create Redis deployment for caching purposes |
|
||||||
`ui.color` | `#34577c` | UI color
|
| `ui.color` | `#34577c` | UI color |
|
||||||
`ui.message` | `None` | UI greetings message
|
| `ui.message` | `None` | UI greetings message |
|
||||||
`ui.logo` | `None` | UI logo
|
| `ui.logo` | `None` | UI logo |
|
||||||
`faults.delay` | `false` | Random HTTP response delays between 0 and 5 seconds
|
| `faults.delay` | `false` | Random HTTP response delays between 0 and 5 seconds |
|
||||||
`faults.error` | `false` | 1/3 chances of a random HTTP response error
|
| `faults.error` | `false` | 1/3 chances of a random HTTP response error |
|
||||||
`faults.unhealthy` | `false` | When set, the healthy state is never reached
|
| `faults.unhealthy` | `false` | When set, the healthy state is never reached |
|
||||||
`faults.unready` | `false` | When set, the ready state is never reached
|
| `faults.unready` | `false` | When set, the ready state is never reached |
|
||||||
`faults.testFail` | `false` | When set, a helm test is included which always fails
|
| `faults.testFail` | `false` | When set, a helm test is included which always fails |
|
||||||
`faults.testTimeout` | `false` | When set, a helm test is included which always times out
|
| `faults.testTimeout` | `false` | When set, a helm test is included which always times out |
|
||||||
`image.repository` | `stefanprodan/podinfo` | Image repository
|
| `image.repository` | `stefanprodan/podinfo` | Image repository |
|
||||||
`image.tag` | `<VERSION>` | Image tag
|
| `image.tag` | `<VERSION>` | Image tag |
|
||||||
`image.pullPolicy` | `IfNotPresent` | Image pull policy
|
| `image.pullPolicy` | `IfNotPresent` | Image pull policy |
|
||||||
`service.enabled` | `true` | Create a Kubernetes Service, should be disabled when using [Flagger](https://flagger.app)
|
| `service.enabled` | `true` | Create a Kubernetes Service, should be disabled when using [Flagger](https://flagger.app) |
|
||||||
`service.type` | `ClusterIP` | Type of the Kubernetes Service
|
| `service.type` | `ClusterIP` | Type of the Kubernetes Service |
|
||||||
`service.metricsPort` | `9797` | Prometheus metrics endpoint port
|
| `service.metricsPort` | `9797` | Prometheus metrics endpoint port |
|
||||||
`service.httpPort` | `9898` | Container HTTP port
|
| `service.httpPort` | `9898` | Container HTTP port |
|
||||||
`service.externalPort` | `9898` | ClusterIP HTTP port
|
| `service.externalPort` | `9898` | ClusterIP HTTP port |
|
||||||
`service.grpcPort` | `9999` | ClusterIP gPRC port
|
| `service.grpcPort` | `9999` | ClusterIP gPRC port |
|
||||||
`service.grpcService` | `podinfo` | gPRC service name
|
| `service.grpcService` | `podinfo` | gPRC service name |
|
||||||
`service.nodePort` | `31198` | NodePort for the HTTP endpoint
|
| `service.nodePort` | `31198` | NodePort for the HTTP endpoint |
|
||||||
`h2c.enabled` | `false` | Allow upgrading to h2c (non-TLS version of HTTP/2)
|
| `h2c.enabled` | `false` | Allow upgrading to h2c (non-TLS version of HTTP/2) |
|
||||||
`hpa.enabled` | `false` | Enables the Kubernetes HPA
|
| `extraEnvs` | `[]` | Extra environment variables for the podinfo container |
|
||||||
`hpa.maxReplicas` | `10` | Maximum amount of pods
|
| `config.path` | `""` | config file path |
|
||||||
`hpa.cpu` | `None` | Target CPU usage per pod
|
| `config.name` | `""` | config file name |
|
||||||
`hpa.memory` | `None` | Target memory usage per pod
|
| `extraArgs` | `[]` | Additional command line arguments to pass to podinfo container |
|
||||||
`hpa.requests` | `None` | Target HTTP requests per second per pod
|
| `hpa.enabled` | `false` | Enables the Kubernetes HPA |
|
||||||
`serviceAccount.enabled` | `false` | Whether a service account should be created
|
| `hpa.maxReplicas` | `10` | Maximum amount of pods |
|
||||||
`serviceAccount.name` | `None` | The name of the service account to use, if not set and create is true, a name is generated using the fullname template
|
| `hpa.cpu` | `None` | Target CPU usage per pod |
|
||||||
`securityContext` | `{}` | The security context to be set on the podinfo container
|
| `hpa.memory` | `None` | Target memory usage per pod |
|
||||||
`linkerd.profile.enabled` | `false` | Create Linkerd service profile
|
| `hpa.requests` | `None` | Target HTTP requests per second per pod |
|
||||||
`serviceMonitor.enabled` | `false` | Whether a Prometheus Operator service monitor should be created
|
| `serviceAccount.enabled` | `false` | Whether a service account should be created |
|
||||||
`serviceMonitor.interval` | `15s` | Prometheus scraping interval
|
| `serviceAccount.name` | `None` | The name of the service account to use, if not set and create is true, a name is generated using the fullname template |
|
||||||
`serviceMonitor.additionalLabels` | `{}` | Add additional labels to the service monitor |
|
| `serviceAccount.imagePullSecrets` | `[]` | List of image pull secrets if pulling from private registries. |
|
||||||
`ingress.enabled` | `false` | Enables Ingress
|
| `securityContext` | `{}` | The security context to be set on the podinfo container |
|
||||||
`ingress.className ` | `""` | Use ingressClassName
|
| `podSecurityContext` | `{}` | The security context to be set on the pod |
|
||||||
`ingress.annotations` | `{}` | Ingress annotations
|
| `linkerd.profile.enabled` | `false` | Create Linkerd service profile |
|
||||||
`ingress.hosts` | `[]` | Ingress accepted hosts
|
| `serviceMonitor.enabled` | `false` | Whether a Prometheus Operator service monitor should be created |
|
||||||
`ingress.tls` | `[]` | Ingress TLS configuration
|
| `serviceMonitor.interval` | `15s` | Prometheus scraping interval |
|
||||||
`resources.requests.cpu` | `1m` | Pod CPU request
|
| `serviceMonitor.additionalLabels` | `{}` | Add additional labels to the service monitor |
|
||||||
`resources.requests.memory` | `16Mi` | Pod memory request
|
| `ingress.enabled` | `false` | Enables Ingress |
|
||||||
`resources.limits.cpu` | `None` | Pod CPU limit
|
| `ingress.className` | `""` | Use ingressClassName |
|
||||||
`resources.limits.memory` | `None` | Pod memory limit
|
| `ingress.additionalLabels` | `{}` | Add additional labels to the ingress |
|
||||||
`nodeSelector` | `{}` | Node labels for pod assignment
|
| `ingress.annotations` | `{}` | Ingress annotations |
|
||||||
`tolerations` | `[]` | List of node taints to tolerate
|
| `ingress.hosts` | `[]` | Ingress accepted hosts |
|
||||||
`affinity` | `None` | Node/pod affinities
|
| `ingress.tls` | `[]` | Ingress TLS configuration |
|
||||||
`podAnnotations` | `{}` | Pod annotations
|
| `resources.requests.cpu` | `1m` | Pod CPU request |
|
||||||
|
| `resources.requests.memory` | `16Mi` | Pod memory request |
|
||||||
|
| `resources.limits.cpu` | `None` | Pod CPU limit |
|
||||||
|
| `resources.limits.memory` | `None` | Pod memory limit |
|
||||||
|
| `nodeSelector` | `{}` | Node labels for pod assignment |
|
||||||
|
| `tolerations` | `[]` | List of node taints to tolerate |
|
||||||
|
| `affinity` | `None` | Node/pod affinities |
|
||||||
|
| `podAnnotations` | `{}` | Pod annotations |
|
||||||
|
|
||||||
Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example,
|
Specify each parameter using the `--set key=value[,key=value]` argument to `helm install`. For example,
|
||||||
|
|
||||||
@@ -110,14 +133,3 @@ $ helm install my-release podinfo/podinfo -f values.yaml
|
|||||||
```
|
```
|
||||||
|
|
||||||
> **Tip**: You can use the default [values.yaml](values.yaml)
|
> **Tip**: You can use the default [values.yaml](values.yaml)
|
||||||
|
|
||||||
## Upgrading the chart
|
|
||||||
|
|
||||||
### To =< 5.0.0
|
|
||||||
|
|
||||||
Version 5.0.0 is a major update.
|
|
||||||
|
|
||||||
* The chart now follows the new Kubernetes label recommendations:
|
|
||||||
<https://kubernetes.io/docs/concepts/overview/working-with-objects/common-labels/>
|
|
||||||
|
|
||||||
The simplest way to update is to do a force upgrade, which recreates the resources by doing a delete and an install.
|
|
||||||
|
|||||||
@@ -6,15 +6,15 @@
|
|||||||
{{- end }}
|
{{- end }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
{{- else if contains "NodePort" .Values.service.type }}
|
{{- else if contains "NodePort" .Values.service.type }}
|
||||||
export NODE_PORT=$(kubectl get --namespace {{ .Release.Namespace }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "podinfo.fullname" . }})
|
export NODE_PORT=$(kubectl get --namespace {{ include "podinfo.namespace" . }} -o jsonpath="{.spec.ports[0].nodePort}" services {{ template "podinfo.fullname" . }})
|
||||||
export NODE_IP=$(kubectl get nodes --namespace {{ .Release.Namespace }} -o jsonpath="{.items[0].status.addresses[0].address}")
|
export NODE_IP=$(kubectl get nodes --namespace {{ include "podinfo.namespace" . }} -o jsonpath="{.items[0].status.addresses[0].address}")
|
||||||
echo http://$NODE_IP:$NODE_PORT
|
echo http://$NODE_IP:$NODE_PORT
|
||||||
{{- else if contains "LoadBalancer" .Values.service.type }}
|
{{- else if contains "LoadBalancer" .Values.service.type }}
|
||||||
NOTE: It may take a few minutes for the LoadBalancer IP to be available.
|
NOTE: It may take a few minutes for the LoadBalancer IP to be available.
|
||||||
You can watch the status of by running 'kubectl get svc -w {{ template "podinfo.fullname" . }}'
|
You can watch the status of by running 'kubectl get svc -w {{ template "podinfo.fullname" . }}'
|
||||||
export SERVICE_IP=$(kubectl get svc --namespace {{ .Release.Namespace }} {{ template "podinfo.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
|
export SERVICE_IP=$(kubectl get svc --namespace {{ include "podinfo.namespace" . }} {{ template "podinfo.fullname" . }} -o jsonpath='{.status.loadBalancer.ingress[0].ip}')
|
||||||
echo http://$SERVICE_IP:{{ .Values.service.externalPort }}
|
echo http://$SERVICE_IP:{{ .Values.service.externalPort }}
|
||||||
{{- else if contains "ClusterIP" .Values.service.type }}
|
{{- else if contains "ClusterIP" .Values.service.type }}
|
||||||
echo "Visit http://127.0.0.1:8080 to use your application"
|
echo "Visit http://127.0.0.1:8080 to use your application"
|
||||||
kubectl -n {{ .Release.Namespace }} port-forward deploy/{{ template "podinfo.fullname" . }} 8080:{{ .Values.service.externalPort }}
|
kubectl -n {{ include "podinfo.namespace" . }} port-forward deploy/{{ template "podinfo.fullname" . }} 8080:{{ .Values.service.externalPort }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
|||||||
@@ -23,6 +23,13 @@ If release name contains chart name it will be used as a full name.
|
|||||||
{{- end }}
|
{{- end }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
|
||||||
|
{{/*
|
||||||
|
Allow the release namespace to be overridden for multi-namespace deployments in combined charts.
|
||||||
|
*/}}
|
||||||
|
{{- define "podinfo.namespace" -}}
|
||||||
|
{{- default .Release.Namespace .Values.namespaceOverride | trunc 63 | trimSuffix "-" -}}
|
||||||
|
{{- end -}}
|
||||||
|
|
||||||
{{/*
|
{{/*
|
||||||
Create chart name and version as used by the chart label.
|
Create chart name and version as used by the chart label.
|
||||||
*/}}
|
*/}}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ apiVersion: cert-manager.io/v1
|
|||||||
kind: Certificate
|
kind: Certificate
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}
|
name: {{ template "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
spec:
|
spec:
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ apiVersion: apps/v1
|
|||||||
kind: Deployment
|
kind: Deployment
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}
|
name: {{ template "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
spec:
|
spec:
|
||||||
@@ -73,7 +74,7 @@ spec:
|
|||||||
{{- if .Values.cache }}
|
{{- if .Values.cache }}
|
||||||
- --cache-server={{ .Values.cache }}
|
- --cache-server={{ .Values.cache }}
|
||||||
{{- else if .Values.redis.enabled }}
|
{{- else if .Values.redis.enabled }}
|
||||||
- --cache-server={{ template "podinfo.fullname" . }}-redis:6379
|
- --cache-server=tcp://{{ template "podinfo.fullname" . }}-redis:6379
|
||||||
{{- end }}
|
{{- end }}
|
||||||
- --level={{ .Values.logLevel }}
|
- --level={{ .Values.logLevel }}
|
||||||
- --random-delay={{ .Values.faults.delay }}
|
- --random-delay={{ .Values.faults.delay }}
|
||||||
@@ -87,6 +88,15 @@ spec:
|
|||||||
{{- if .Values.h2c.enabled }}
|
{{- if .Values.h2c.enabled }}
|
||||||
- --h2c
|
- --h2c
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
{{- with .Values.config.path }}
|
||||||
|
- --config-path={{ . }}
|
||||||
|
{{- end }}
|
||||||
|
{{- with .Values.config.name }}
|
||||||
|
- --config={{ . }}
|
||||||
|
{{- end }}
|
||||||
|
{{- with .Values.extraArgs }}
|
||||||
|
{{- toYaml . | nindent 12 }}
|
||||||
|
{{- end }}
|
||||||
env:
|
env:
|
||||||
{{- if .Values.ui.message }}
|
{{- if .Values.ui.message }}
|
||||||
- name: PODINFO_UI_MESSAGE
|
- name: PODINFO_UI_MESSAGE
|
||||||
@@ -104,6 +114,9 @@ spec:
|
|||||||
- name: PODINFO_BACKEND_URL
|
- name: PODINFO_BACKEND_URL
|
||||||
value: {{ .Values.backend }}
|
value: {{ .Values.backend }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
{{- if .Values.extraEnvs }}
|
||||||
|
{{ toYaml .Values.extraEnvs | indent 10 }}
|
||||||
|
{{- end }}
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
containerPort: {{ .Values.service.httpPort | default 9898 }}
|
containerPort: {{ .Values.service.httpPort | default 9898 }}
|
||||||
@@ -129,6 +142,22 @@ spec:
|
|||||||
containerPort: {{ .Values.service.grpcPort }}
|
containerPort: {{ .Values.service.grpcPort }}
|
||||||
protocol: TCP
|
protocol: TCP
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
{{- if .Values.probes.startup.enable }}
|
||||||
|
startupProbe:
|
||||||
|
exec:
|
||||||
|
command:
|
||||||
|
- podcli
|
||||||
|
- check
|
||||||
|
- http
|
||||||
|
- localhost:{{ .Values.service.httpPort | default 9898 }}/healthz
|
||||||
|
{{- with .Values.probes.startup }}
|
||||||
|
initialDelaySeconds: {{ .initialDelaySeconds | default 1 }}
|
||||||
|
timeoutSeconds: {{ .timeoutSeconds | default 5 }}
|
||||||
|
failureThreshold: {{ .failureThreshold | default 3 }}
|
||||||
|
successThreshold: {{ .successThreshold | default 1 }}
|
||||||
|
periodSeconds: {{ .periodSeconds | default 10 }}
|
||||||
|
{{- end }}
|
||||||
|
{{- end }}
|
||||||
livenessProbe:
|
livenessProbe:
|
||||||
exec:
|
exec:
|
||||||
command:
|
command:
|
||||||
@@ -136,8 +165,13 @@ spec:
|
|||||||
- check
|
- check
|
||||||
- http
|
- http
|
||||||
- localhost:{{ .Values.service.httpPort | default 9898 }}/healthz
|
- localhost:{{ .Values.service.httpPort | default 9898 }}/healthz
|
||||||
initialDelaySeconds: 1
|
{{- with .Values.probes.liveness }}
|
||||||
timeoutSeconds: 5
|
initialDelaySeconds: {{ .initialDelaySeconds | default 1 }}
|
||||||
|
timeoutSeconds: {{ .timeoutSeconds | default 5 }}
|
||||||
|
failureThreshold: {{ .failureThreshold | default 3 }}
|
||||||
|
successThreshold: {{ .successThreshold | default 1 }}
|
||||||
|
periodSeconds: {{ .periodSeconds | default 10 }}
|
||||||
|
{{- end }}
|
||||||
readinessProbe:
|
readinessProbe:
|
||||||
exec:
|
exec:
|
||||||
command:
|
command:
|
||||||
@@ -145,8 +179,13 @@ spec:
|
|||||||
- check
|
- check
|
||||||
- http
|
- http
|
||||||
- localhost:{{ .Values.service.httpPort | default 9898 }}/readyz
|
- localhost:{{ .Values.service.httpPort | default 9898 }}/readyz
|
||||||
initialDelaySeconds: 1
|
{{- with .Values.probes.readiness }}
|
||||||
timeoutSeconds: 5
|
initialDelaySeconds: {{ .initialDelaySeconds | default 1 }}
|
||||||
|
timeoutSeconds: {{ .timeoutSeconds | default 5 }}
|
||||||
|
failureThreshold: {{ .failureThreshold | default 3 }}
|
||||||
|
successThreshold: {{ .successThreshold | default 1 }}
|
||||||
|
periodSeconds: {{ .periodSeconds | default 10 }}
|
||||||
|
{{- end }}
|
||||||
volumeMounts:
|
volumeMounts:
|
||||||
- name: data
|
- name: data
|
||||||
mountPath: /data
|
mountPath: /data
|
||||||
@@ -157,6 +196,10 @@ spec:
|
|||||||
{{- end }}
|
{{- end }}
|
||||||
resources:
|
resources:
|
||||||
{{ toYaml .Values.resources | indent 12 }}
|
{{ toYaml .Values.resources | indent 12 }}
|
||||||
|
{{- with .Values.podSecurityContext }}
|
||||||
|
securityContext:
|
||||||
|
{{ toYaml . | indent 8 }}
|
||||||
|
{{- end }}
|
||||||
{{- with .Values.nodeSelector }}
|
{{- with .Values.nodeSelector }}
|
||||||
nodeSelector:
|
nodeSelector:
|
||||||
{{ toYaml . | indent 8 }}
|
{{ toYaml . | indent 8 }}
|
||||||
@@ -177,3 +220,7 @@ spec:
|
|||||||
secret:
|
secret:
|
||||||
secretName: {{ template "podinfo.tlsSecretName" . }}
|
secretName: {{ template "podinfo.tlsSecretName" . }}
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
{{- with .Values.topologySpreadConstraints }}
|
||||||
|
topologySpreadConstraints:
|
||||||
|
{{- toYaml . | nindent 8 }}
|
||||||
|
{{- end }}
|
||||||
@@ -1,8 +1,9 @@
|
|||||||
{{- if .Values.hpa.enabled -}}
|
{{- if .Values.hpa.enabled -}}
|
||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}
|
name: {{ template "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
spec:
|
spec:
|
||||||
|
|||||||
@@ -5,8 +5,12 @@ apiVersion: networking.k8s.io/v1
|
|||||||
kind: Ingress
|
kind: Ingress
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ $fullName }}
|
name: {{ $fullName }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
|
{{- with .Values.ingress.additionalLabels }}
|
||||||
|
{{- toYaml . | nindent 4 }}
|
||||||
|
{{- end }}
|
||||||
{{- with .Values.ingress.annotations }}
|
{{- with .Values.ingress.annotations }}
|
||||||
annotations:
|
annotations:
|
||||||
{{- toYaml . | nindent 4 }}
|
{{- toYaml . | nindent 4 }}
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
apiVersion: linkerd.io/v1alpha2
|
apiVersion: linkerd.io/v1alpha2
|
||||||
kind: ServiceProfile
|
kind: ServiceProfile
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}.svc.cluster.local
|
name: {{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}.svc.cluster.local
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
spec:
|
spec:
|
||||||
|
|||||||
14
charts/podinfo/templates/pdb.yaml
Normal file
14
charts/podinfo/templates/pdb.yaml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
{{- if and .Values.podDisruptionBudget (gt (int .Values.replicaCount) 1) }}
|
||||||
|
apiVersion: policy/v1
|
||||||
|
kind: PodDisruptionBudget
|
||||||
|
metadata:
|
||||||
|
name: {{ include "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
|
labels:
|
||||||
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
|
spec:
|
||||||
|
selector:
|
||||||
|
matchLabels:
|
||||||
|
{{- include "podinfo.selectorLabels" . | nindent 6 }}
|
||||||
|
{{- toYaml .Values.podDisruptionBudget | nindent 2 }}
|
||||||
|
{{- end }}
|
||||||
@@ -3,6 +3,7 @@ apiVersion: v1
|
|||||||
kind: Service
|
kind: Service
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}
|
name: {{ template "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
{{- with .Values.service.annotations }}
|
{{- with .Values.service.annotations }}
|
||||||
|
|||||||
@@ -5,4 +5,8 @@ metadata:
|
|||||||
name: {{ template "podinfo.serviceAccountName" . }}
|
name: {{ template "podinfo.serviceAccountName" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
|
{{- with .Values.serviceAccount.imagePullSecrets }}
|
||||||
|
imagePullSecrets:
|
||||||
|
{{- toYaml . | nindent 2 }}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
{{- end -}}
|
||||||
@@ -3,6 +3,7 @@ apiVersion: monitoring.coreos.com/v1
|
|||||||
kind: ServiceMonitor
|
kind: ServiceMonitor
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}
|
name: {{ template "podinfo.fullname" . }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
{{- with .Values.serviceMonitor.additionalLabels }}
|
{{- with .Values.serviceMonitor.additionalLabels }}
|
||||||
@@ -15,7 +16,7 @@ spec:
|
|||||||
interval: {{ .Values.serviceMonitor.interval }}
|
interval: {{ .Values.serviceMonitor.interval }}
|
||||||
namespaceSelector:
|
namespaceSelector:
|
||||||
matchNames:
|
matchNames:
|
||||||
- {{ .Release.Namespace }}
|
- {{ include "podinfo.namespace" . }}
|
||||||
selector:
|
selector:
|
||||||
matchLabels:
|
matchLabels:
|
||||||
{{- include "podinfo.selectorLabels" . | nindent 6 }}
|
{{- include "podinfo.selectorLabels" . | nindent 6 }}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-cache-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-cache-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
@@ -24,6 +25,6 @@ spec:
|
|||||||
curl -s -XDELETE ${PODINFO_SVC}/cache/test
|
curl -s -XDELETE ${PODINFO_SVC}/cache/test
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_SVC
|
- name: PODINFO_SVC
|
||||||
value: "{{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}:{{ .Values.service.externalPort }}"
|
value: "{{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}:{{ .Values.service.externalPort }}"
|
||||||
restartPolicy: Never
|
restartPolicy: Never
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-fault-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-fault-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-grpc-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-grpc-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
@@ -15,5 +16,5 @@ spec:
|
|||||||
- name: grpc-health-probe
|
- name: grpc-health-probe
|
||||||
image: stefanprodan/grpc_health_probe:v0.3.0
|
image: stefanprodan/grpc_health_probe:v0.3.0
|
||||||
command: ['grpc_health_probe']
|
command: ['grpc_health_probe']
|
||||||
args: ['-addr={{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}:{{ .Values.service.grpcPort }}']
|
args: ['-addr={{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}:{{ .Values.service.grpcPort }}']
|
||||||
restartPolicy: Never
|
restartPolicy: Never
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-jwt-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-jwt-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
@@ -22,5 +23,5 @@ spec:
|
|||||||
curl -sH "Authorization: Bearer ${TOKEN}" ${PODINFO_SVC}/token/validate | grep test
|
curl -sH "Authorization: Bearer ${TOKEN}" ${PODINFO_SVC}/token/validate | grep test
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_SVC
|
- name: PODINFO_SVC
|
||||||
value: "{{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}:{{ .Values.service.externalPort }}"
|
value: "{{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}:{{ .Values.service.externalPort }}"
|
||||||
restartPolicy: Never
|
restartPolicy: Never
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-service-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-service-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
@@ -21,5 +22,5 @@ spec:
|
|||||||
curl -s ${PODINFO_SVC}/api/info | grep version
|
curl -s ${PODINFO_SVC}/api/info | grep version
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_SVC
|
- name: PODINFO_SVC
|
||||||
value: "{{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}:{{ .Values.service.externalPort }}"
|
value: "{{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}:{{ .Values.service.externalPort }}"
|
||||||
restartPolicy: Never
|
restartPolicy: Never
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-fault-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-fault-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ apiVersion: v1
|
|||||||
kind: Pod
|
kind: Pod
|
||||||
metadata:
|
metadata:
|
||||||
name: {{ template "podinfo.fullname" . }}-tls-test-{{ randAlphaNum 5 | lower }}
|
name: {{ template "podinfo.fullname" . }}-tls-test-{{ randAlphaNum 5 | lower }}
|
||||||
|
namespace: {{ include "podinfo.namespace" . }}
|
||||||
labels:
|
labels:
|
||||||
{{- include "podinfo.labels" . | nindent 4 }}
|
{{- include "podinfo.labels" . | nindent 4 }}
|
||||||
annotations:
|
annotations:
|
||||||
@@ -22,6 +23,6 @@ spec:
|
|||||||
curl -sk ${PODINFO_SVC}/api/info | grep version
|
curl -sk ${PODINFO_SVC}/api/info | grep version
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_SVC
|
- name: PODINFO_SVC
|
||||||
value: "https://{{ template "podinfo.fullname" . }}.{{ .Release.Namespace }}:{{ .Values.tls.port }}"
|
value: "https://{{ template "podinfo.fullname" . }}.{{ include "podinfo.namespace" . }}:{{ .Values.tls.port }}"
|
||||||
restartPolicy: Never
|
restartPolicy: Never
|
||||||
{{- end }}
|
{{- end }}
|
||||||
@@ -8,7 +8,7 @@ backends: []
|
|||||||
|
|
||||||
image:
|
image:
|
||||||
repository: ghcr.io/stefanprodan/podinfo
|
repository: ghcr.io/stefanprodan/podinfo
|
||||||
tag: 6.1.1
|
tag: 6.9.2
|
||||||
pullPolicy: IfNotPresent
|
pullPolicy: IfNotPresent
|
||||||
|
|
||||||
ui:
|
ui:
|
||||||
@@ -41,6 +41,16 @@ service:
|
|||||||
h2c:
|
h2c:
|
||||||
enabled: false
|
enabled: false
|
||||||
|
|
||||||
|
# config file settings
|
||||||
|
config:
|
||||||
|
# config file path
|
||||||
|
path: ""
|
||||||
|
# config file name
|
||||||
|
name: ""
|
||||||
|
|
||||||
|
# Additional command line arguments to pass to podinfo container
|
||||||
|
extraArgs: []
|
||||||
|
|
||||||
# enable tls on the podinfo service
|
# enable tls on the podinfo service
|
||||||
tls:
|
tls:
|
||||||
enabled: false
|
enabled: false
|
||||||
@@ -77,13 +87,13 @@ hpa:
|
|||||||
# average http requests per second per pod (k8s-prometheus-adapter)
|
# average http requests per second per pod (k8s-prometheus-adapter)
|
||||||
requests:
|
requests:
|
||||||
|
|
||||||
# Redis address in the format <host>:<port>
|
# Redis address in the format tcp://<host>:<port>
|
||||||
cache: ""
|
cache: ""
|
||||||
# Redis deployment
|
# Redis deployment
|
||||||
redis:
|
redis:
|
||||||
enabled: true
|
enabled: true
|
||||||
repository: redis
|
repository: redis
|
||||||
tag: 6.0.8
|
tag: 7.0.7
|
||||||
|
|
||||||
serviceAccount:
|
serviceAccount:
|
||||||
# Specifies whether a service account should be created
|
# Specifies whether a service account should be created
|
||||||
@@ -91,13 +101,19 @@ serviceAccount:
|
|||||||
# The name of the service account to use.
|
# The name of the service account to use.
|
||||||
# If not set and create is true, a name is generated using the fullname template
|
# If not set and create is true, a name is generated using the fullname template
|
||||||
name:
|
name:
|
||||||
|
# List of image pull secrets if pulling from private registries
|
||||||
|
imagePullSecrets: []
|
||||||
|
|
||||||
# set container security context
|
# set container security context
|
||||||
securityContext: {}
|
securityContext: {}
|
||||||
|
|
||||||
|
# set pod security context
|
||||||
|
podSecurityContext: {}
|
||||||
|
|
||||||
ingress:
|
ingress:
|
||||||
enabled: false
|
enabled: false
|
||||||
className: ""
|
className: ""
|
||||||
|
additionalLabels: {}
|
||||||
annotations: {}
|
annotations: {}
|
||||||
# kubernetes.io/ingress.class: nginx
|
# kubernetes.io/ingress.class: nginx
|
||||||
# kubernetes.io/tls-acme: "true"
|
# kubernetes.io/tls-acme: "true"
|
||||||
@@ -128,6 +144,14 @@ resources:
|
|||||||
cpu: 100m
|
cpu: 100m
|
||||||
memory: 64Mi
|
memory: 64Mi
|
||||||
|
|
||||||
|
# Extra environment variables for the podinfo container
|
||||||
|
extraEnvs: []
|
||||||
|
# Example on how to configure extraEnvs
|
||||||
|
# - name: OTEL_EXPORTER_OTLP_TRACES_ENDPOINT
|
||||||
|
# value: "http://otel:4317"
|
||||||
|
# - name: MULTIPLE_VALUES
|
||||||
|
# value: TEST
|
||||||
|
|
||||||
nodeSelector: {}
|
nodeSelector: {}
|
||||||
|
|
||||||
tolerations: []
|
tolerations: []
|
||||||
@@ -135,3 +159,25 @@ tolerations: []
|
|||||||
affinity: {}
|
affinity: {}
|
||||||
|
|
||||||
podAnnotations: {}
|
podAnnotations: {}
|
||||||
|
|
||||||
|
# https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes
|
||||||
|
probes:
|
||||||
|
readiness:
|
||||||
|
initialDelaySeconds: 1
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 3
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
liveness:
|
||||||
|
initialDelaySeconds: 1
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 3
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
startup:
|
||||||
|
enable: false
|
||||||
|
initialDelaySeconds: 10
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 20
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ backends: []
|
|||||||
|
|
||||||
image:
|
image:
|
||||||
repository: ghcr.io/stefanprodan/podinfo
|
repository: ghcr.io/stefanprodan/podinfo
|
||||||
tag: 6.1.1
|
tag: 6.9.2
|
||||||
pullPolicy: IfNotPresent
|
pullPolicy: IfNotPresent
|
||||||
|
|
||||||
ui:
|
ui:
|
||||||
@@ -45,6 +45,16 @@ service:
|
|||||||
h2c:
|
h2c:
|
||||||
enabled: false
|
enabled: false
|
||||||
|
|
||||||
|
# config file settings
|
||||||
|
config:
|
||||||
|
# config file path
|
||||||
|
path: ""
|
||||||
|
# config file name
|
||||||
|
name: ""
|
||||||
|
|
||||||
|
# Additional command line arguments to pass to podinfo container
|
||||||
|
extraArgs: []
|
||||||
|
|
||||||
# enable tls on the podinfo service
|
# enable tls on the podinfo service
|
||||||
tls:
|
tls:
|
||||||
enabled: false
|
enabled: false
|
||||||
@@ -81,13 +91,13 @@ hpa:
|
|||||||
# average http requests per second per pod (k8s-prometheus-adapter)
|
# average http requests per second per pod (k8s-prometheus-adapter)
|
||||||
requests:
|
requests:
|
||||||
|
|
||||||
# Redis address in the format <host>:<port>
|
# Redis address in the format tcp://<host>:<port>
|
||||||
cache: ""
|
cache: ""
|
||||||
# Redis deployment
|
# Redis deployment
|
||||||
redis:
|
redis:
|
||||||
enabled: false
|
enabled: false
|
||||||
repository: redis
|
repository: redis
|
||||||
tag: 6.0.8
|
tag: 7.0.7
|
||||||
|
|
||||||
serviceAccount:
|
serviceAccount:
|
||||||
# Specifies whether a service account should be created
|
# Specifies whether a service account should be created
|
||||||
@@ -95,13 +105,19 @@ serviceAccount:
|
|||||||
# The name of the service account to use.
|
# The name of the service account to use.
|
||||||
# If not set and create is true, a name is generated using the fullname template
|
# If not set and create is true, a name is generated using the fullname template
|
||||||
name:
|
name:
|
||||||
|
# List of image pull secrets if pulling from private registries
|
||||||
|
imagePullSecrets: []
|
||||||
|
|
||||||
# set container security context
|
# set container security context
|
||||||
securityContext: {}
|
securityContext: {}
|
||||||
|
|
||||||
|
# set pod security context
|
||||||
|
podSecurityContext: {}
|
||||||
|
|
||||||
ingress:
|
ingress:
|
||||||
enabled: false
|
enabled: false
|
||||||
className: ""
|
className: ""
|
||||||
|
additionalLabels: {}
|
||||||
annotations: {}
|
annotations: {}
|
||||||
# kubernetes.io/ingress.class: nginx
|
# kubernetes.io/ingress.class: nginx
|
||||||
# kubernetes.io/tls-acme: "true"
|
# kubernetes.io/tls-acme: "true"
|
||||||
@@ -131,6 +147,14 @@ resources:
|
|||||||
cpu: 1m
|
cpu: 1m
|
||||||
memory: 16Mi
|
memory: 16Mi
|
||||||
|
|
||||||
|
# Extra environment variables for the podinfo container
|
||||||
|
extraEnvs: []
|
||||||
|
# Example on how to configure extraEnvs
|
||||||
|
# - name: OTEL_EXPORTER_OTLP_TRACES_ENDPOINT
|
||||||
|
# value: "http://otel:4317"
|
||||||
|
# - name: MULTIPLE_VALUES
|
||||||
|
# value: TEST
|
||||||
|
|
||||||
nodeSelector: {}
|
nodeSelector: {}
|
||||||
|
|
||||||
tolerations: []
|
tolerations: []
|
||||||
@@ -138,3 +162,32 @@ tolerations: []
|
|||||||
affinity: {}
|
affinity: {}
|
||||||
|
|
||||||
podAnnotations: {}
|
podAnnotations: {}
|
||||||
|
|
||||||
|
# https://kubernetes.io/docs/concepts/workloads/pods/pod-topology-spread-constraints/
|
||||||
|
topologySpreadConstraints: []
|
||||||
|
|
||||||
|
# Disruption budget will be configured only when the replicaCount is greater than 1
|
||||||
|
podDisruptionBudget: {}
|
||||||
|
# maxUnavailable: 1
|
||||||
|
|
||||||
|
# https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle#container-probes
|
||||||
|
probes:
|
||||||
|
readiness:
|
||||||
|
initialDelaySeconds: 1
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 3
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
liveness:
|
||||||
|
initialDelaySeconds: 1
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 3
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
startup:
|
||||||
|
enable: false
|
||||||
|
initialDelaySeconds: 10
|
||||||
|
timeoutSeconds: 5
|
||||||
|
failureThreshold: 20
|
||||||
|
successThreshold: 1
|
||||||
|
periodSeconds: 10
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strconv"
|
"strconv"
|
||||||
@@ -14,10 +13,11 @@ import (
|
|||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"go.uber.org/zap/zapcore"
|
"go.uber.org/zap/zapcore"
|
||||||
|
|
||||||
"github.com/stefanprodan/podinfo/pkg/api"
|
"github.com/stefanprodan/podinfo/pkg/api/grpc"
|
||||||
"github.com/stefanprodan/podinfo/pkg/grpc"
|
"github.com/stefanprodan/podinfo/pkg/api/http"
|
||||||
"github.com/stefanprodan/podinfo/pkg/signals"
|
"github.com/stefanprodan/podinfo/pkg/signals"
|
||||||
"github.com/stefanprodan/podinfo/pkg/version"
|
"github.com/stefanprodan/podinfo/pkg/version"
|
||||||
|
go_grpc "google.golang.org/grpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
@@ -33,7 +33,7 @@ func main() {
|
|||||||
fs.StringSlice("backend-url", []string{}, "backend service URL")
|
fs.StringSlice("backend-url", []string{}, "backend service URL")
|
||||||
fs.Duration("http-client-timeout", 2*time.Minute, "client timeout duration")
|
fs.Duration("http-client-timeout", 2*time.Minute, "client timeout duration")
|
||||||
fs.Duration("http-server-timeout", 30*time.Second, "server read and write timeout duration")
|
fs.Duration("http-server-timeout", 30*time.Second, "server read and write timeout duration")
|
||||||
fs.Duration("http-server-shutdown-timeout", 5*time.Second, "server graceful shutdown timeout duration")
|
fs.Duration("server-shutdown-timeout", 5*time.Second, "server graceful shutdown timeout duration")
|
||||||
fs.String("data-path", "/data", "data local path")
|
fs.String("data-path", "/data", "data local path")
|
||||||
fs.String("config-path", "", "config dir path")
|
fs.String("config-path", "", "config dir path")
|
||||||
fs.String("cert-path", "/data/cert", "certificate path for HTTPS port")
|
fs.String("cert-path", "/data/cert", "certificate path for HTTPS port")
|
||||||
@@ -52,7 +52,7 @@ func main() {
|
|||||||
fs.Bool("unready", false, "when set, ready state is never reached")
|
fs.Bool("unready", false, "when set, ready state is never reached")
|
||||||
fs.Int("stress-cpu", 0, "number of CPU cores with 100 load")
|
fs.Int("stress-cpu", 0, "number of CPU cores with 100 load")
|
||||||
fs.Int("stress-memory", 0, "MB of data to load into memory")
|
fs.Int("stress-memory", 0, "MB of data to load into memory")
|
||||||
fs.String("cache-server", "", "Redis address in the format <host>:<port>")
|
fs.String("cache-server", "", "Redis address in the format 'tcp://<host>:<port>'")
|
||||||
fs.String("otel-service-name", "", "service name for reporting to open telemetry address, when not set tracing is disabled")
|
fs.String("otel-service-name", "", "service name for reporting to open telemetry address, when not set tracing is disabled")
|
||||||
|
|
||||||
versionFlag := fs.BoolP("version", "v", false, "get version number")
|
versionFlag := fs.BoolP("version", "v", false, "get version number")
|
||||||
@@ -135,13 +135,16 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// start gRPC server
|
// start gRPC server
|
||||||
|
var grpcServer *go_grpc.Server
|
||||||
if grpcCfg.Port > 0 {
|
if grpcCfg.Port > 0 {
|
||||||
grpcSrv, _ := grpc.NewServer(&grpcCfg, logger)
|
grpcSrv, _ := grpc.NewServer(&grpcCfg, logger)
|
||||||
go grpcSrv.ListenAndServe()
|
//grpcinfoSrv, _ := grpc.NewInfoServer(&grpcCfg)
|
||||||
|
|
||||||
|
grpcServer = grpcSrv.ListenAndServe()
|
||||||
}
|
}
|
||||||
|
|
||||||
// load HTTP server config
|
// load HTTP server config
|
||||||
var srvCfg api.Config
|
var srvCfg http.Config
|
||||||
if err := viper.Unmarshal(&srvCfg); err != nil {
|
if err := viper.Unmarshal(&srvCfg); err != nil {
|
||||||
logger.Panic("config unmarshal failed", zap.Error(err))
|
logger.Panic("config unmarshal failed", zap.Error(err))
|
||||||
}
|
}
|
||||||
@@ -154,9 +157,13 @@ func main() {
|
|||||||
)
|
)
|
||||||
|
|
||||||
// start HTTP server
|
// start HTTP server
|
||||||
srv, _ := api.NewServer(&srvCfg, logger)
|
srv, _ := http.NewServer(&srvCfg, logger)
|
||||||
|
httpServer, httpsServer, healthy, ready := srv.ListenAndServe()
|
||||||
|
|
||||||
|
// graceful shutdown
|
||||||
stopCh := signals.SetupSignalHandler()
|
stopCh := signals.SetupSignalHandler()
|
||||||
srv.ListenAndServe(stopCh)
|
sd, _ := signals.NewShutdown(srvCfg.ServerShutdownTimeout, logger)
|
||||||
|
sd.Graceful(stopCh, httpServer, httpsServer, grpcServer, healthy, ready)
|
||||||
}
|
}
|
||||||
|
|
||||||
func initZap(logLevel string) (*zap.Logger, error) {
|
func initZap(logLevel string) (*zap.Logger, error) {
|
||||||
@@ -238,12 +245,12 @@ func beginStressTest(cpus int, mem int, logger *zap.Logger) {
|
|||||||
logger.Error("memory stress failed", zap.Error(err))
|
logger.Error("memory stress failed", zap.Error(err))
|
||||||
}
|
}
|
||||||
|
|
||||||
stressMemoryPayload, err = ioutil.ReadFile(path)
|
stressMemoryPayload, err = os.ReadFile(path)
|
||||||
f.Close()
|
f.Close()
|
||||||
os.Remove(path)
|
os.Remove(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("memory stress failed", zap.Error(err))
|
logger.Error("memory stress failed", zap.Error(err))
|
||||||
}
|
}
|
||||||
logger.Info("starting CPU stress", zap.Int("memory", len(stressMemoryPayload)))
|
logger.Info("starting MEMORY stress", zap.Int("memory", len(stressMemoryPayload)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: backend
|
- name: backend
|
||||||
image: ghcr.io/stefanprodan/podinfo:6.1.1
|
image: ghcr.io/stefanprodan/podinfo:6.9.2
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
@@ -42,7 +42,7 @@ spec:
|
|||||||
- --grpc-port=9999
|
- --grpc-port=9999
|
||||||
- --grpc-service-name=backend
|
- --grpc-service-name=backend
|
||||||
- --level=info
|
- --level=info
|
||||||
- --cache-server=cache:6379
|
- --cache-server=tcp://cache:6379
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_UI_COLOR
|
- name: PODINFO_UI_COLOR
|
||||||
value: "#34577c"
|
value: "#34577c"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: backend
|
name: backend
|
||||||
|
|||||||
2
deploy/bases/cache/deployment.yaml
vendored
2
deploy/bases/cache/deployment.yaml
vendored
@@ -13,7 +13,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: redis
|
- name: redis
|
||||||
image: redis:6.0.1
|
image: redis:7.0.7
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
command:
|
command:
|
||||||
- redis-server
|
- redis-server
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: frontend
|
- name: frontend
|
||||||
image: ghcr.io/stefanprodan/podinfo:6.1.1
|
image: ghcr.io/stefanprodan/podinfo:6.9.2
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
@@ -41,7 +41,7 @@ spec:
|
|||||||
- --port-metrics=9797
|
- --port-metrics=9797
|
||||||
- --level=info
|
- --level=info
|
||||||
- --backend-url=http://backend:9898/echo
|
- --backend-url=http://backend:9898/echo
|
||||||
- --cache-server=cache:6379
|
- --cache-server=tcp://cache:6379
|
||||||
env:
|
env:
|
||||||
- name: PODINFO_UI_COLOR
|
- name: PODINFO_UI_COLOR
|
||||||
value: "#34577c"
|
value: "#34577c"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: frontend
|
name: frontend
|
||||||
|
|||||||
@@ -4,4 +4,6 @@ resources:
|
|||||||
- service.yaml
|
- service.yaml
|
||||||
- deployment.yaml
|
- deployment.yaml
|
||||||
- hpa.yaml
|
- hpa.yaml
|
||||||
|
images:
|
||||||
|
- name: ghcr.io/stefanprodan/podinfo
|
||||||
|
newTag: 6.9.2
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: backend
|
name: backend
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: frontend
|
name: frontend
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ spec:
|
|||||||
serviceAccountName: webapp
|
serviceAccountName: webapp
|
||||||
containers:
|
containers:
|
||||||
- name: backend
|
- name: backend
|
||||||
image: ghcr.io/stefanprodan/podinfo:6.1.1
|
image: ghcr.io/stefanprodan/podinfo:6.9.2
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: backend
|
name: backend
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ spec:
|
|||||||
serviceAccountName: webapp
|
serviceAccountName: webapp
|
||||||
containers:
|
containers:
|
||||||
- name: frontend
|
- name: frontend
|
||||||
image: ghcr.io/stefanprodan/podinfo:6.1.1
|
image: ghcr.io/stefanprodan/podinfo:6.9.2
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: frontend
|
name: frontend
|
||||||
|
|||||||
141
go.mod
141
go.mod
@@ -1,84 +1,79 @@
|
|||||||
module github.com/stefanprodan/podinfo
|
module github.com/stefanprodan/podinfo
|
||||||
|
|
||||||
go 1.17
|
go 1.25
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751
|
github.com/chzyer/readline v1.5.1
|
||||||
github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e
|
github.com/fatih/color v1.18.0
|
||||||
github.com/dgrijalva/jwt-go/v4 v4.0.0-preview1
|
github.com/fsnotify/fsnotify v1.9.0
|
||||||
github.com/fatih/color v1.9.0
|
github.com/golang-jwt/jwt/v4 v4.5.2
|
||||||
github.com/fsnotify/fsnotify v1.4.9
|
github.com/gomodule/redigo v1.9.2
|
||||||
github.com/gomodule/redigo v1.8.4
|
github.com/gorilla/mux v1.8.1
|
||||||
github.com/gorilla/mux v1.8.0
|
github.com/gorilla/websocket v1.5.3
|
||||||
github.com/gorilla/websocket v1.4.2
|
github.com/prometheus/client_golang v1.22.0
|
||||||
github.com/prometheus/client_golang v1.11.0
|
github.com/spf13/cobra v1.10.1
|
||||||
github.com/spf13/cobra v1.2.1
|
github.com/spf13/pflag v1.0.10
|
||||||
github.com/spf13/pflag v1.0.5
|
github.com/spf13/viper v1.21.0
|
||||||
github.com/spf13/viper v1.8.1
|
github.com/swaggo/http-swagger v1.3.4
|
||||||
github.com/swaggo/http-swagger v1.0.0
|
github.com/swaggo/swag v1.16.4
|
||||||
github.com/swaggo/swag v1.7.6
|
go.opentelemetry.io/contrib/instrumentation/github.com/gorilla/mux/otelmux v0.63.0
|
||||||
go.opentelemetry.io/contrib/instrumentation/github.com/gorilla/mux/otelmux v0.28.0
|
go.opentelemetry.io/contrib/instrumentation/net/http/httptrace/otelhttptrace v0.63.0
|
||||||
go.opentelemetry.io/contrib/instrumentation/net/http/httptrace/otelhttptrace v0.28.0
|
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.63.0
|
||||||
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.28.0
|
go.opentelemetry.io/contrib/propagators/aws v1.38.0
|
||||||
go.opentelemetry.io/contrib/propagators/aws v1.3.0
|
go.opentelemetry.io/contrib/propagators/b3 v1.38.0
|
||||||
go.opentelemetry.io/contrib/propagators/b3 v1.3.0
|
go.opentelemetry.io/contrib/propagators/jaeger v1.38.0
|
||||||
go.opentelemetry.io/contrib/propagators/jaeger v1.3.0
|
go.opentelemetry.io/contrib/propagators/ot v1.38.0
|
||||||
go.opentelemetry.io/contrib/propagators/ot v1.3.0
|
go.opentelemetry.io/otel v1.38.0
|
||||||
go.opentelemetry.io/otel v1.3.0
|
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.38.0
|
||||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.3.0
|
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.38.0
|
||||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.3.0
|
go.opentelemetry.io/otel/sdk v1.38.0
|
||||||
go.opentelemetry.io/otel/sdk v1.3.0
|
go.opentelemetry.io/otel/trace v1.38.0
|
||||||
go.opentelemetry.io/otel/trace v1.3.0
|
go.uber.org/zap v1.27.0
|
||||||
go.uber.org/zap v1.19.1
|
golang.org/x/net v0.44.0
|
||||||
golang.org/x/net v0.0.0-20211216030914-fe4d6282115f
|
google.golang.org/grpc v1.75.1
|
||||||
google.golang.org/grpc v1.43.0
|
google.golang.org/protobuf v1.36.9
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/KyleBanks/depth v1.2.1 // indirect
|
github.com/KyleBanks/depth v1.2.1 // indirect
|
||||||
github.com/PuerkitoBio/purell v1.1.1 // indirect
|
|
||||||
github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 // indirect
|
|
||||||
github.com/beorn7/perks v1.0.1 // indirect
|
github.com/beorn7/perks v1.0.1 // indirect
|
||||||
github.com/cenkalti/backoff/v4 v4.1.2 // indirect
|
github.com/cenkalti/backoff/v5 v5.0.3 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.1.1 // indirect
|
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||||
github.com/felixge/httpsnoop v1.0.2 // indirect
|
github.com/felixge/httpsnoop v1.0.4 // indirect
|
||||||
github.com/go-logr/logr v1.2.1 // indirect
|
github.com/go-logr/logr v1.4.3 // indirect
|
||||||
github.com/go-logr/stdr v1.2.0 // indirect
|
github.com/go-logr/stdr v1.2.2 // indirect
|
||||||
github.com/go-openapi/jsonpointer v0.19.5 // indirect
|
github.com/go-openapi/jsonpointer v0.21.1 // indirect
|
||||||
github.com/go-openapi/jsonreference v0.19.5 // indirect
|
github.com/go-openapi/jsonreference v0.21.0 // indirect
|
||||||
github.com/go-openapi/spec v0.20.3 // indirect
|
github.com/go-openapi/spec v0.21.0 // indirect
|
||||||
github.com/go-openapi/swag v0.19.14 // indirect
|
github.com/go-openapi/swag v0.23.1 // indirect
|
||||||
github.com/golang/protobuf v1.5.2 // indirect
|
github.com/go-viper/mapstructure/v2 v2.4.0 // indirect
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0 // indirect
|
github.com/google/uuid v1.6.0 // indirect
|
||||||
github.com/hashicorp/hcl v1.0.0 // indirect
|
github.com/grpc-ecosystem/grpc-gateway/v2 v2.27.2 // indirect
|
||||||
github.com/inconshreveable/mousetrap v1.0.0 // indirect
|
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||||
github.com/josharian/intern v1.0.0 // indirect
|
github.com/josharian/intern v1.0.0 // indirect
|
||||||
github.com/magiconair/properties v1.8.5 // indirect
|
github.com/mailru/easyjson v0.9.0 // indirect
|
||||||
github.com/mailru/easyjson v0.7.6 // indirect
|
github.com/mattn/go-colorable v0.1.14 // indirect
|
||||||
github.com/mattn/go-colorable v0.1.4 // indirect
|
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||||
github.com/mattn/go-isatty v0.0.12 // indirect
|
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1 // indirect
|
github.com/pelletier/go-toml/v2 v2.2.4 // indirect
|
||||||
github.com/mitchellh/mapstructure v1.4.1 // indirect
|
github.com/prometheus/client_model v0.6.1 // indirect
|
||||||
github.com/pelletier/go-toml v1.9.3 // indirect
|
github.com/prometheus/common v0.62.0 // indirect
|
||||||
github.com/prometheus/client_model v0.2.0 // indirect
|
github.com/prometheus/procfs v0.15.1 // indirect
|
||||||
github.com/prometheus/common v0.26.0 // indirect
|
github.com/sagikazarmark/locafero v0.11.0 // indirect
|
||||||
github.com/prometheus/procfs v0.6.0 // indirect
|
github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 // indirect
|
||||||
github.com/spf13/afero v1.6.0 // indirect
|
github.com/spf13/afero v1.15.0 // indirect
|
||||||
github.com/spf13/cast v1.3.1 // indirect
|
github.com/spf13/cast v1.10.0 // indirect
|
||||||
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
github.com/subosito/gotenv v1.6.0 // indirect
|
||||||
github.com/subosito/gotenv v1.2.0 // indirect
|
github.com/swaggo/files v1.0.1 // indirect
|
||||||
github.com/swaggo/files v0.0.0-20190704085106-630677cd5c14 // indirect
|
go.opentelemetry.io/auto/sdk v1.1.0 // indirect
|
||||||
go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.3.0 // indirect
|
go.opentelemetry.io/otel/metric v1.38.0 // indirect
|
||||||
go.opentelemetry.io/otel/internal/metric v0.26.0 // indirect
|
go.opentelemetry.io/proto/otlp v1.7.1 // indirect
|
||||||
go.opentelemetry.io/otel/metric v0.26.0 // indirect
|
go.uber.org/multierr v1.11.0 // indirect
|
||||||
go.opentelemetry.io/proto/otlp v0.11.0 // indirect
|
go.yaml.in/yaml/v3 v3.0.4 // indirect
|
||||||
go.uber.org/atomic v1.7.0 // indirect
|
golang.org/x/sys v0.36.0 // indirect
|
||||||
go.uber.org/multierr v1.6.0 // indirect
|
golang.org/x/text v0.29.0 // indirect
|
||||||
golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40 // indirect
|
golang.org/x/tools v0.36.0 // indirect
|
||||||
golang.org/x/text v0.3.6 // indirect
|
google.golang.org/genproto/googleapis/api v0.0.0-20250825161204-c5933d9347a5 // indirect
|
||||||
golang.org/x/tools v0.1.5 // indirect
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250825161204-c5933d9347a5 // indirect
|
||||||
google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
google.golang.org/protobuf v1.27.1 // indirect
|
|
||||||
gopkg.in/ini.v1 v1.62.0 // indirect
|
|
||||||
gopkg.in/yaml.v2 v2.4.0 // indirect
|
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: podinfod
|
- name: podinfod
|
||||||
image: ghcr.io/stefanprodan/podinfo:6.1.1
|
image: ghcr.io/stefanprodan/podinfo:6.9.2
|
||||||
imagePullPolicy: IfNotPresent
|
imagePullPolicy: IfNotPresent
|
||||||
ports:
|
ports:
|
||||||
- name: http
|
- name: http
|
||||||
@@ -72,3 +72,9 @@ spec:
|
|||||||
requests:
|
requests:
|
||||||
cpu: 100m
|
cpu: 100m
|
||||||
memory: 64Mi
|
memory: 64Mi
|
||||||
|
volumeMounts:
|
||||||
|
- name: data
|
||||||
|
mountPath: /data
|
||||||
|
volumes:
|
||||||
|
- name: data
|
||||||
|
emptyDir: {}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
apiVersion: autoscaling/v2beta2
|
apiVersion: autoscaling/v2
|
||||||
kind: HorizontalPodAutoscaler
|
kind: HorizontalPodAutoscaler
|
||||||
metadata:
|
metadata:
|
||||||
name: podinfo
|
name: podinfo
|
||||||
|
|||||||
@@ -4,4 +4,3 @@ resources:
|
|||||||
- hpa.yaml
|
- hpa.yaml
|
||||||
- deployment.yaml
|
- deployment.yaml
|
||||||
- service.yaml
|
- service.yaml
|
||||||
|
|
||||||
|
|||||||
@@ -1,34 +0,0 @@
|
|||||||
package api
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestEchoHandler(t *testing.T) {
|
|
||||||
expected := `{"test": true}`
|
|
||||||
req, err := http.NewRequest("POST", "/api/echo", strings.NewReader(expected))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
rr := httptest.NewRecorder()
|
|
||||||
srv := NewMockServer()
|
|
||||||
handler := http.HandlerFunc(srv.echoHandler)
|
|
||||||
|
|
||||||
handler.ServeHTTP(rr, req)
|
|
||||||
|
|
||||||
// Check the status code is what we expect.
|
|
||||||
if status := rr.Code; status != http.StatusAccepted {
|
|
||||||
t.Errorf("handler returned wrong status code: got %v want %v",
|
|
||||||
status, http.StatusAccepted)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check the response body is what we expect.
|
|
||||||
if rr.Body.String() != expected {
|
|
||||||
t.Fatalf("handler returned unexpected body:\ngot \n%v \nwant \n%s",
|
|
||||||
rr.Body.String(), expected)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
21
pkg/api/grpc/delay.go
Normal file
21
pkg/api/grpc/delay.go
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/delay"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DelayServer struct {
|
||||||
|
pb.UnimplementedDelayServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *DelayServer) Delay(ctx context.Context, delayInput *pb.DelayRequest) (*pb.DelayResponse, error) {
|
||||||
|
|
||||||
|
time.Sleep(time.Duration(delayInput.Seconds) * time.Second)
|
||||||
|
return &pb.DelayResponse{Message: delayInput.Seconds}, nil
|
||||||
|
}
|
||||||
211
pkg/api/grpc/delay/delay.pb.go
Normal file
211
pkg/api/grpc/delay/delay.pb.go
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.28.1
|
||||||
|
// protoc v4.25.0
|
||||||
|
// source: delay/delay.proto
|
||||||
|
|
||||||
|
package delay
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type DelayRequest struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
Seconds int64 `protobuf:"varint,1,opt,name=seconds,proto3" json:"seconds,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayRequest) Reset() {
|
||||||
|
*x = DelayRequest{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_delay_delay_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayRequest) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*DelayRequest) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *DelayRequest) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_delay_delay_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use DelayRequest.ProtoReflect.Descriptor instead.
|
||||||
|
func (*DelayRequest) Descriptor() ([]byte, []int) {
|
||||||
|
return file_delay_delay_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayRequest) GetSeconds() int64 {
|
||||||
|
if x != nil {
|
||||||
|
return x.Seconds
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
type DelayResponse struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
Message int64 `protobuf:"varint,1,opt,name=message,proto3" json:"message,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayResponse) Reset() {
|
||||||
|
*x = DelayResponse{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_delay_delay_proto_msgTypes[1]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayResponse) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*DelayResponse) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *DelayResponse) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_delay_delay_proto_msgTypes[1]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use DelayResponse.ProtoReflect.Descriptor instead.
|
||||||
|
func (*DelayResponse) Descriptor() ([]byte, []int) {
|
||||||
|
return file_delay_delay_proto_rawDescGZIP(), []int{1}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *DelayResponse) GetMessage() int64 {
|
||||||
|
if x != nil {
|
||||||
|
return x.Message
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_delay_delay_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_delay_delay_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x11, 0x64, 0x65, 0x6c, 0x61, 0x79, 0x2f, 0x64, 0x65, 0x6c, 0x61, 0x79, 0x2e, 0x70, 0x72,
|
||||||
|
0x6f, 0x74, 0x6f, 0x12, 0x05, 0x64, 0x65, 0x6c, 0x61, 0x79, 0x22, 0x28, 0x0a, 0x0c, 0x44, 0x65,
|
||||||
|
0x6c, 0x61, 0x79, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x18, 0x0a, 0x07, 0x73, 0x65,
|
||||||
|
0x63, 0x6f, 0x6e, 0x64, 0x73, 0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, 0x73, 0x65, 0x63,
|
||||||
|
0x6f, 0x6e, 0x64, 0x73, 0x22, 0x29, 0x0a, 0x0d, 0x44, 0x65, 0x6c, 0x61, 0x79, 0x52, 0x65, 0x73,
|
||||||
|
0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x6d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65,
|
||||||
|
0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, 0x6d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x32,
|
||||||
|
0x44, 0x0a, 0x0c, 0x44, 0x65, 0x6c, 0x61, 0x79, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12,
|
||||||
|
0x34, 0x0a, 0x05, 0x44, 0x65, 0x6c, 0x61, 0x79, 0x12, 0x13, 0x2e, 0x64, 0x65, 0x6c, 0x61, 0x79,
|
||||||
|
0x2e, 0x44, 0x65, 0x6c, 0x61, 0x79, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x14, 0x2e,
|
||||||
|
0x64, 0x65, 0x6c, 0x61, 0x79, 0x2e, 0x44, 0x65, 0x6c, 0x61, 0x79, 0x52, 0x65, 0x73, 0x70, 0x6f,
|
||||||
|
0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x09, 0x5a, 0x07, 0x2e, 0x2f, 0x64, 0x65, 0x6c, 0x61, 0x79,
|
||||||
|
0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_delay_delay_proto_rawDescOnce sync.Once
|
||||||
|
file_delay_delay_proto_rawDescData = file_delay_delay_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_delay_delay_proto_rawDescGZIP() []byte {
|
||||||
|
file_delay_delay_proto_rawDescOnce.Do(func() {
|
||||||
|
file_delay_delay_proto_rawDescData = protoimpl.X.CompressGZIP(file_delay_delay_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_delay_delay_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_delay_delay_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
|
||||||
|
var file_delay_delay_proto_goTypes = []interface{}{
|
||||||
|
(*DelayRequest)(nil), // 0: delay.DelayRequest
|
||||||
|
(*DelayResponse)(nil), // 1: delay.DelayResponse
|
||||||
|
}
|
||||||
|
var file_delay_delay_proto_depIdxs = []int32{
|
||||||
|
0, // 0: delay.DelayService.Delay:input_type -> delay.DelayRequest
|
||||||
|
1, // 1: delay.DelayService.Delay:output_type -> delay.DelayResponse
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_delay_delay_proto_init() }
|
||||||
|
func file_delay_delay_proto_init() {
|
||||||
|
if File_delay_delay_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_delay_delay_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*DelayRequest); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file_delay_delay_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*DelayResponse); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_delay_delay_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 2,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_delay_delay_proto_goTypes,
|
||||||
|
DependencyIndexes: file_delay_delay_proto_depIdxs,
|
||||||
|
MessageInfos: file_delay_delay_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_delay_delay_proto = out.File
|
||||||
|
file_delay_delay_proto_rawDesc = nil
|
||||||
|
file_delay_delay_proto_goTypes = nil
|
||||||
|
file_delay_delay_proto_depIdxs = nil
|
||||||
|
}
|
||||||
17
pkg/api/grpc/delay/delay.proto
Normal file
17
pkg/api/grpc/delay/delay.proto
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./delay";
|
||||||
|
|
||||||
|
package delay;
|
||||||
|
|
||||||
|
message DelayRequest {
|
||||||
|
int64 seconds = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message DelayResponse {
|
||||||
|
int64 message = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
service DelayService {
|
||||||
|
rpc Delay (DelayRequest) returns (DelayResponse) {}
|
||||||
|
}
|
||||||
105
pkg/api/grpc/delay/delay_grpc.pb.go
Normal file
105
pkg/api/grpc/delay/delay_grpc.pb.go
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.2.0
|
||||||
|
// - protoc v4.25.0
|
||||||
|
// source: delay/delay.proto
|
||||||
|
|
||||||
|
package delay
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
// DelayServiceClient is the client API for DelayService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type DelayServiceClient interface {
|
||||||
|
Delay(ctx context.Context, in *DelayRequest, opts ...grpc.CallOption) (*DelayResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type delayServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewDelayServiceClient(cc grpc.ClientConnInterface) DelayServiceClient {
|
||||||
|
return &delayServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *delayServiceClient) Delay(ctx context.Context, in *DelayRequest, opts ...grpc.CallOption) (*DelayResponse, error) {
|
||||||
|
out := new(DelayResponse)
|
||||||
|
err := c.cc.Invoke(ctx, "/delay.DelayService/Delay", in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DelayServiceServer is the server API for DelayService service.
|
||||||
|
// All implementations must embed UnimplementedDelayServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type DelayServiceServer interface {
|
||||||
|
Delay(context.Context, *DelayRequest) (*DelayResponse, error)
|
||||||
|
mustEmbedUnimplementedDelayServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedDelayServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedDelayServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedDelayServiceServer) Delay(context.Context, *DelayRequest) (*DelayResponse, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Delay not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedDelayServiceServer) mustEmbedUnimplementedDelayServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafeDelayServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to DelayServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafeDelayServiceServer interface {
|
||||||
|
mustEmbedUnimplementedDelayServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterDelayServiceServer(s grpc.ServiceRegistrar, srv DelayServiceServer) {
|
||||||
|
s.RegisterService(&DelayService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _DelayService_Delay_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(DelayRequest)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(DelayServiceServer).Delay(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: "/delay.DelayService/Delay",
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(DelayServiceServer).Delay(ctx, req.(*DelayRequest))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DelayService_ServiceDesc is the grpc.ServiceDesc for DelayService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var DelayService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "delay.DelayService",
|
||||||
|
HandlerType: (*DelayServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Delay",
|
||||||
|
Handler: _DelayService_Delay_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "delay/delay.proto",
|
||||||
|
}
|
||||||
75
pkg/api/grpc/delay_test.go
Normal file
75
pkg/api/grpc/delay_test.go
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/delay"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGrpcDelay(t *testing.T) {
|
||||||
|
|
||||||
|
// Server initialization
|
||||||
|
// bufconn => uses in-memory connection instead of system network I/O
|
||||||
|
lis := bufconn.Listen(1024 * 1024)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
lis.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
t.Cleanup(func() {
|
||||||
|
srv.Stop()
|
||||||
|
})
|
||||||
|
|
||||||
|
delay.RegisterDelayServiceServer(srv, &DelayServer{})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(lis); err != nil {
|
||||||
|
log.Fatalf("srv.Serve %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
// - Test
|
||||||
|
dialer := func(context.Context, string) (net.Conn, error) {
|
||||||
|
return lis.Dial()
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(ctx, "", grpc.WithContextDialer(dialer), grpc.WithInsecure())
|
||||||
|
t.Cleanup(func() {
|
||||||
|
conn.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("grpc.DialContext %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := delay.NewDelayServiceClient(conn)
|
||||||
|
res, err := client.Delay(context.Background(), &delay.DelayRequest{Seconds: 3})
|
||||||
|
|
||||||
|
// Check the status code is what we expect.
|
||||||
|
if _, ok := status.FromError(err); !ok {
|
||||||
|
t.Errorf("Delay returned type %T, want %T", err, status.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if res != nil {
|
||||||
|
fmt.Printf("res %v\n", res)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the response body is what we expect. Here we expect the response to be "3" as the delay is set to 3 seconds.
|
||||||
|
expected := "3"
|
||||||
|
r := regexp.MustCompile(expected)
|
||||||
|
if !r.MatchString(strconv.FormatInt(res.Message, 10)) {
|
||||||
|
t.Fatalf("Returned unexpected body:\ngot \n%v \nwant \n%s",
|
||||||
|
res.Message, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
20
pkg/api/grpc/echo.go
Normal file
20
pkg/api/grpc/echo.go
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/echo"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
type echoServer struct {
|
||||||
|
echo.UnimplementedEchoServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *echoServer) Echo(ctx context.Context, message *echo.Message) (*echo.Message, error) {
|
||||||
|
|
||||||
|
s.logger.Info("Received message body from client:", zap.String("input body", message.Body))
|
||||||
|
return &echo.Message{Body: message.Body}, nil
|
||||||
|
}
|
||||||
146
pkg/api/grpc/echo/echo.pb.go
Normal file
146
pkg/api/grpc/echo/echo.pb.go
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.26.0
|
||||||
|
// protoc v4.24.3
|
||||||
|
// source: echo/echo.proto
|
||||||
|
|
||||||
|
package echo
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type Message struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
Body string `protobuf:"bytes,1,opt,name=body,proto3" json:"body,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Message) Reset() {
|
||||||
|
*x = Message{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_echo_echo_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Message) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*Message) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *Message) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_echo_echo_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use Message.ProtoReflect.Descriptor instead.
|
||||||
|
func (*Message) Descriptor() ([]byte, []int) {
|
||||||
|
return file_echo_echo_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *Message) GetBody() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Body
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_echo_echo_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_echo_echo_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x0f, 0x65, 0x63, 0x68, 0x6f, 0x2f, 0x65, 0x63, 0x68, 0x6f, 0x2e, 0x70, 0x72, 0x6f, 0x74,
|
||||||
|
0x6f, 0x12, 0x04, 0x65, 0x63, 0x68, 0x6f, 0x22, 0x1d, 0x0a, 0x07, 0x4d, 0x65, 0x73, 0x73, 0x61,
|
||||||
|
0x67, 0x65, 0x12, 0x12, 0x0a, 0x04, 0x62, 0x6f, 0x64, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09,
|
||||||
|
0x52, 0x04, 0x62, 0x6f, 0x64, 0x79, 0x32, 0x35, 0x0a, 0x0b, 0x45, 0x63, 0x68, 0x6f, 0x53, 0x65,
|
||||||
|
0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x26, 0x0a, 0x04, 0x45, 0x63, 0x68, 0x6f, 0x12, 0x0d, 0x2e,
|
||||||
|
0x65, 0x63, 0x68, 0x6f, 0x2e, 0x4d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x1a, 0x0d, 0x2e, 0x65,
|
||||||
|
0x63, 0x68, 0x6f, 0x2e, 0x4d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x22, 0x00, 0x42, 0x08, 0x5a,
|
||||||
|
0x06, 0x2e, 0x2f, 0x65, 0x63, 0x68, 0x6f, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_echo_echo_proto_rawDescOnce sync.Once
|
||||||
|
file_echo_echo_proto_rawDescData = file_echo_echo_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_echo_echo_proto_rawDescGZIP() []byte {
|
||||||
|
file_echo_echo_proto_rawDescOnce.Do(func() {
|
||||||
|
file_echo_echo_proto_rawDescData = protoimpl.X.CompressGZIP(file_echo_echo_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_echo_echo_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_echo_echo_proto_msgTypes = make([]protoimpl.MessageInfo, 1)
|
||||||
|
var file_echo_echo_proto_goTypes = []interface{}{
|
||||||
|
(*Message)(nil), // 0: echo.Message
|
||||||
|
}
|
||||||
|
var file_echo_echo_proto_depIdxs = []int32{
|
||||||
|
0, // 0: echo.EchoService.Echo:input_type -> echo.Message
|
||||||
|
0, // 1: echo.EchoService.Echo:output_type -> echo.Message
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_echo_echo_proto_init() }
|
||||||
|
func file_echo_echo_proto_init() {
|
||||||
|
if File_echo_echo_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_echo_echo_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*Message); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_echo_echo_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 1,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_echo_echo_proto_goTypes,
|
||||||
|
DependencyIndexes: file_echo_echo_proto_depIdxs,
|
||||||
|
MessageInfos: file_echo_echo_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_echo_echo_proto = out.File
|
||||||
|
file_echo_echo_proto_rawDesc = nil
|
||||||
|
file_echo_echo_proto_goTypes = nil
|
||||||
|
file_echo_echo_proto_depIdxs = nil
|
||||||
|
}
|
||||||
14
pkg/api/grpc/echo/echo.proto
Normal file
14
pkg/api/grpc/echo/echo.proto
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./echo";
|
||||||
|
|
||||||
|
package echo;
|
||||||
|
|
||||||
|
message Message {
|
||||||
|
string body = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
service EchoService {
|
||||||
|
rpc Echo(Message) returns (Message) {}
|
||||||
|
}
|
||||||
109
pkg/api/grpc/echo/echo_grpc.pb.go
Normal file
109
pkg/api/grpc/echo/echo_grpc.pb.go
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.3.0
|
||||||
|
// - protoc v4.24.3
|
||||||
|
// source: echo/echo.proto
|
||||||
|
|
||||||
|
package echo
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
const (
|
||||||
|
EchoService_Echo_FullMethodName = "/echo.EchoService/Echo"
|
||||||
|
)
|
||||||
|
|
||||||
|
// EchoServiceClient is the client API for EchoService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type EchoServiceClient interface {
|
||||||
|
Echo(ctx context.Context, in *Message, opts ...grpc.CallOption) (*Message, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type echoServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewEchoServiceClient(cc grpc.ClientConnInterface) EchoServiceClient {
|
||||||
|
return &echoServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *echoServiceClient) Echo(ctx context.Context, in *Message, opts ...grpc.CallOption) (*Message, error) {
|
||||||
|
out := new(Message)
|
||||||
|
err := c.cc.Invoke(ctx, EchoService_Echo_FullMethodName, in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EchoServiceServer is the server API for EchoService service.
|
||||||
|
// All implementations must embed UnimplementedEchoServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type EchoServiceServer interface {
|
||||||
|
Echo(context.Context, *Message) (*Message, error)
|
||||||
|
mustEmbedUnimplementedEchoServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedEchoServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedEchoServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedEchoServiceServer) Echo(context.Context, *Message) (*Message, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Echo not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedEchoServiceServer) mustEmbedUnimplementedEchoServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafeEchoServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to EchoServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafeEchoServiceServer interface {
|
||||||
|
mustEmbedUnimplementedEchoServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterEchoServiceServer(s grpc.ServiceRegistrar, srv EchoServiceServer) {
|
||||||
|
s.RegisterService(&EchoService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _EchoService_Echo_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(Message)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(EchoServiceServer).Echo(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: EchoService_Echo_FullMethodName,
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(EchoServiceServer).Echo(ctx, req.(*Message))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EchoService_ServiceDesc is the grpc.ServiceDesc for EchoService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var EchoService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "echo.EchoService",
|
||||||
|
HandlerType: (*EchoServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Echo",
|
||||||
|
Handler: _EchoService_Echo_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "echo/echo.proto",
|
||||||
|
}
|
||||||
65
pkg/api/grpc/echo_test.go
Normal file
65
pkg/api/grpc/echo_test.go
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"regexp"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/echo"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGrpcEcho(t *testing.T) {
|
||||||
|
|
||||||
|
lis := bufconn.Listen(1024 * 1024)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
lis.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
s := NewMockGrpcServer()
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
t.Cleanup(func() {
|
||||||
|
srv.Stop()
|
||||||
|
})
|
||||||
|
|
||||||
|
echo.RegisterEchoServiceServer(srv, &echoServer{config: s.config, logger: s.logger})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(lis); err != nil {
|
||||||
|
log.Fatalf("srv.Serve %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
dialer := func(context.Context, string) (net.Conn, error) {
|
||||||
|
return lis.Dial()
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(ctx, "", grpc.WithContextDialer(dialer), grpc.WithInsecure())
|
||||||
|
t.Cleanup(func() {
|
||||||
|
conn.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("grpc.DialContext %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := echo.NewEchoServiceClient(conn)
|
||||||
|
res, err := client.Echo(context.Background(), &echo.Message{Body: "test123-test"})
|
||||||
|
|
||||||
|
if _, ok := status.FromError(err); !ok {
|
||||||
|
t.Errorf("Echo returned type %T, want %T", err, status.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := ".*body.*test123-test.*"
|
||||||
|
r := regexp.MustCompile(expected)
|
||||||
|
if !r.MatchString(res.String()) {
|
||||||
|
t.Fatalf("Returned unexpected body:\ngot \n%v \nwant \n%s",
|
||||||
|
res, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
19
pkg/api/grpc/env.go
Normal file
19
pkg/api/grpc/env.go
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/env"
|
||||||
|
)
|
||||||
|
|
||||||
|
type EnvServer struct {
|
||||||
|
pb.UnimplementedEnvServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *EnvServer) Env(ctx context.Context, envInput *pb.EnvRequest) (*pb.EnvResponse, error) {
|
||||||
|
return &pb.EnvResponse{EnvVars: os.Environ()}, nil
|
||||||
|
}
|
||||||
199
pkg/api/grpc/env/env.pb.go
vendored
Normal file
199
pkg/api/grpc/env/env.pb.go
vendored
Normal file
@@ -0,0 +1,199 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.28.1
|
||||||
|
// protoc v4.25.0
|
||||||
|
// source: env/env.proto
|
||||||
|
|
||||||
|
package env
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type EnvRequest struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *EnvRequest) Reset() {
|
||||||
|
*x = EnvRequest{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_env_env_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *EnvRequest) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*EnvRequest) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *EnvRequest) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_env_env_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use EnvRequest.ProtoReflect.Descriptor instead.
|
||||||
|
func (*EnvRequest) Descriptor() ([]byte, []int) {
|
||||||
|
return file_env_env_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
type EnvResponse struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
EnvVars []string `protobuf:"bytes,1,rep,name=envVars,proto3" json:"envVars,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *EnvResponse) Reset() {
|
||||||
|
*x = EnvResponse{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_env_env_proto_msgTypes[1]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *EnvResponse) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*EnvResponse) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *EnvResponse) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_env_env_proto_msgTypes[1]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use EnvResponse.ProtoReflect.Descriptor instead.
|
||||||
|
func (*EnvResponse) Descriptor() ([]byte, []int) {
|
||||||
|
return file_env_env_proto_rawDescGZIP(), []int{1}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *EnvResponse) GetEnvVars() []string {
|
||||||
|
if x != nil {
|
||||||
|
return x.EnvVars
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_env_env_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_env_env_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x0d, 0x65, 0x6e, 0x76, 0x2f, 0x65, 0x6e, 0x76, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12,
|
||||||
|
0x03, 0x65, 0x6e, 0x76, 0x22, 0x0c, 0x0a, 0x0a, 0x45, 0x6e, 0x76, 0x52, 0x65, 0x71, 0x75, 0x65,
|
||||||
|
0x73, 0x74, 0x22, 0x27, 0x0a, 0x0b, 0x45, 0x6e, 0x76, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73,
|
||||||
|
0x65, 0x12, 0x18, 0x0a, 0x07, 0x65, 0x6e, 0x76, 0x56, 0x61, 0x72, 0x73, 0x18, 0x01, 0x20, 0x03,
|
||||||
|
0x28, 0x09, 0x52, 0x07, 0x65, 0x6e, 0x76, 0x56, 0x61, 0x72, 0x73, 0x32, 0x38, 0x0a, 0x0a, 0x45,
|
||||||
|
0x6e, 0x76, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x2a, 0x0a, 0x03, 0x45, 0x6e, 0x76,
|
||||||
|
0x12, 0x0f, 0x2e, 0x65, 0x6e, 0x76, 0x2e, 0x45, 0x6e, 0x76, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73,
|
||||||
|
0x74, 0x1a, 0x10, 0x2e, 0x65, 0x6e, 0x76, 0x2e, 0x45, 0x6e, 0x76, 0x52, 0x65, 0x73, 0x70, 0x6f,
|
||||||
|
0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x07, 0x5a, 0x05, 0x2e, 0x2f, 0x65, 0x6e, 0x76, 0x62, 0x06,
|
||||||
|
0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_env_env_proto_rawDescOnce sync.Once
|
||||||
|
file_env_env_proto_rawDescData = file_env_env_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_env_env_proto_rawDescGZIP() []byte {
|
||||||
|
file_env_env_proto_rawDescOnce.Do(func() {
|
||||||
|
file_env_env_proto_rawDescData = protoimpl.X.CompressGZIP(file_env_env_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_env_env_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_env_env_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
|
||||||
|
var file_env_env_proto_goTypes = []interface{}{
|
||||||
|
(*EnvRequest)(nil), // 0: env.EnvRequest
|
||||||
|
(*EnvResponse)(nil), // 1: env.EnvResponse
|
||||||
|
}
|
||||||
|
var file_env_env_proto_depIdxs = []int32{
|
||||||
|
0, // 0: env.EnvService.Env:input_type -> env.EnvRequest
|
||||||
|
1, // 1: env.EnvService.Env:output_type -> env.EnvResponse
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_env_env_proto_init() }
|
||||||
|
func file_env_env_proto_init() {
|
||||||
|
if File_env_env_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_env_env_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*EnvRequest); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file_env_env_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*EnvResponse); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_env_env_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 2,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_env_env_proto_goTypes,
|
||||||
|
DependencyIndexes: file_env_env_proto_depIdxs,
|
||||||
|
MessageInfos: file_env_env_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_env_env_proto = out.File
|
||||||
|
file_env_env_proto_rawDesc = nil
|
||||||
|
file_env_env_proto_goTypes = nil
|
||||||
|
file_env_env_proto_depIdxs = nil
|
||||||
|
}
|
||||||
15
pkg/api/grpc/env/env.proto
vendored
Normal file
15
pkg/api/grpc/env/env.proto
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./env";
|
||||||
|
|
||||||
|
package env;
|
||||||
|
|
||||||
|
message EnvRequest {}
|
||||||
|
|
||||||
|
message EnvResponse {
|
||||||
|
repeated string envVars = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
service EnvService {
|
||||||
|
rpc Env (EnvRequest) returns (EnvResponse) {}
|
||||||
|
}
|
||||||
105
pkg/api/grpc/env/env_grpc.pb.go
vendored
Normal file
105
pkg/api/grpc/env/env_grpc.pb.go
vendored
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.2.0
|
||||||
|
// - protoc v4.25.0
|
||||||
|
// source: env/env.proto
|
||||||
|
|
||||||
|
package env
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
// EnvServiceClient is the client API for EnvService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type EnvServiceClient interface {
|
||||||
|
Env(ctx context.Context, in *EnvRequest, opts ...grpc.CallOption) (*EnvResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type envServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewEnvServiceClient(cc grpc.ClientConnInterface) EnvServiceClient {
|
||||||
|
return &envServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *envServiceClient) Env(ctx context.Context, in *EnvRequest, opts ...grpc.CallOption) (*EnvResponse, error) {
|
||||||
|
out := new(EnvResponse)
|
||||||
|
err := c.cc.Invoke(ctx, "/env.EnvService/Env", in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnvServiceServer is the server API for EnvService service.
|
||||||
|
// All implementations must embed UnimplementedEnvServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type EnvServiceServer interface {
|
||||||
|
Env(context.Context, *EnvRequest) (*EnvResponse, error)
|
||||||
|
mustEmbedUnimplementedEnvServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedEnvServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedEnvServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedEnvServiceServer) Env(context.Context, *EnvRequest) (*EnvResponse, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Env not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedEnvServiceServer) mustEmbedUnimplementedEnvServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafeEnvServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to EnvServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafeEnvServiceServer interface {
|
||||||
|
mustEmbedUnimplementedEnvServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterEnvServiceServer(s grpc.ServiceRegistrar, srv EnvServiceServer) {
|
||||||
|
s.RegisterService(&EnvService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _EnvService_Env_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(EnvRequest)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(EnvServiceServer).Env(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: "/env.EnvService/Env",
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(EnvServiceServer).Env(ctx, req.(*EnvRequest))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EnvService_ServiceDesc is the grpc.ServiceDesc for EnvService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var EnvService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "env.EnvService",
|
||||||
|
HandlerType: (*EnvServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Env",
|
||||||
|
Handler: _EnvService_Env_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "env/env.proto",
|
||||||
|
}
|
||||||
64
pkg/api/grpc/env_test.go
Normal file
64
pkg/api/grpc/env_test.go
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"regexp"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/env"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGrpcEnv(t *testing.T) {
|
||||||
|
|
||||||
|
lis := bufconn.Listen(1024 * 1024)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
lis.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
t.Cleanup(func() {
|
||||||
|
srv.Stop()
|
||||||
|
})
|
||||||
|
|
||||||
|
env.RegisterEnvServiceServer(srv, &EnvServer{})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(lis); err != nil {
|
||||||
|
log.Fatalf("srv.Serve %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
dialer := func(context.Context, string) (net.Conn, error) {
|
||||||
|
return lis.Dial()
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(ctx, "", grpc.WithContextDialer(dialer), grpc.WithInsecure())
|
||||||
|
t.Cleanup(func() {
|
||||||
|
conn.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("grpc.DialContext %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := env.NewEnvServiceClient(conn)
|
||||||
|
res, err := client.Env(context.Background(), &env.EnvRequest{})
|
||||||
|
|
||||||
|
if _, ok := status.FromError(err); !ok {
|
||||||
|
t.Errorf("Env returned type %T, want %T", err, status.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := ".*PATH.*"
|
||||||
|
r := regexp.MustCompile(expected)
|
||||||
|
if !r.MatchString(res.String()) {
|
||||||
|
t.Fatalf("Returned unexpected body:\ngot \n%v \nwant \n%s",
|
||||||
|
res, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
34
pkg/api/grpc/headers.go
Normal file
34
pkg/api/grpc/headers.go
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/headers"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HeaderServer struct {
|
||||||
|
pb.UnimplementedHeaderServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *HeaderServer) Header(ctx context.Context, in *pb.HeaderRequest) (*pb.HeaderResponse, error) {
|
||||||
|
md, ok := metadata.FromIncomingContext(ctx)
|
||||||
|
if !ok {
|
||||||
|
return nil, status.Errorf(codes.DataLoss, "UnaryEcho: failed to get metadata")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creating slices beacause echoing the header metadata can't be predetermined by the proto contract
|
||||||
|
res := []string{}
|
||||||
|
for i, e := range md {
|
||||||
|
res = append(res, i+"="+strings.Join(e, ","))
|
||||||
|
}
|
||||||
|
|
||||||
|
return &pb.HeaderResponse{Headers: res}, nil
|
||||||
|
|
||||||
|
}
|
||||||
201
pkg/api/grpc/headers/headers.pb.go
Normal file
201
pkg/api/grpc/headers/headers.pb.go
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.28.1
|
||||||
|
// protoc v4.25.0
|
||||||
|
// source: headers/headers.proto
|
||||||
|
|
||||||
|
package header
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type HeaderRequest struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *HeaderRequest) Reset() {
|
||||||
|
*x = HeaderRequest{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_headers_headers_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *HeaderRequest) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*HeaderRequest) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *HeaderRequest) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_headers_headers_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use HeaderRequest.ProtoReflect.Descriptor instead.
|
||||||
|
func (*HeaderRequest) Descriptor() ([]byte, []int) {
|
||||||
|
return file_headers_headers_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
type HeaderResponse struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
Headers []string `protobuf:"bytes,1,rep,name=headers,proto3" json:"headers,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *HeaderResponse) Reset() {
|
||||||
|
*x = HeaderResponse{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_headers_headers_proto_msgTypes[1]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *HeaderResponse) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*HeaderResponse) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *HeaderResponse) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_headers_headers_proto_msgTypes[1]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use HeaderResponse.ProtoReflect.Descriptor instead.
|
||||||
|
func (*HeaderResponse) Descriptor() ([]byte, []int) {
|
||||||
|
return file_headers_headers_proto_rawDescGZIP(), []int{1}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *HeaderResponse) GetHeaders() []string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Headers
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_headers_headers_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_headers_headers_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x15, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x73, 0x2f, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72,
|
||||||
|
0x73, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x06, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x22,
|
||||||
|
0x0f, 0x0a, 0x0d, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74,
|
||||||
|
0x22, 0x2a, 0x0a, 0x0e, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e,
|
||||||
|
0x73, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x73, 0x18, 0x01, 0x20,
|
||||||
|
0x03, 0x28, 0x09, 0x52, 0x07, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x73, 0x32, 0x4a, 0x0a, 0x0d,
|
||||||
|
0x48, 0x65, 0x61, 0x64, 0x65, 0x72, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x39, 0x0a,
|
||||||
|
0x06, 0x48, 0x65, 0x61, 0x64, 0x65, 0x72, 0x12, 0x15, 0x2e, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72,
|
||||||
|
0x2e, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x16,
|
||||||
|
0x2e, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x2e, 0x68, 0x65, 0x61, 0x64, 0x65, 0x72, 0x52, 0x65,
|
||||||
|
0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x0a, 0x5a, 0x08, 0x2e, 0x2f, 0x68, 0x65,
|
||||||
|
0x61, 0x64, 0x65, 0x72, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_headers_headers_proto_rawDescOnce sync.Once
|
||||||
|
file_headers_headers_proto_rawDescData = file_headers_headers_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_headers_headers_proto_rawDescGZIP() []byte {
|
||||||
|
file_headers_headers_proto_rawDescOnce.Do(func() {
|
||||||
|
file_headers_headers_proto_rawDescData = protoimpl.X.CompressGZIP(file_headers_headers_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_headers_headers_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_headers_headers_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
|
||||||
|
var file_headers_headers_proto_goTypes = []interface{}{
|
||||||
|
(*HeaderRequest)(nil), // 0: header.headerRequest
|
||||||
|
(*HeaderResponse)(nil), // 1: header.headerResponse
|
||||||
|
}
|
||||||
|
var file_headers_headers_proto_depIdxs = []int32{
|
||||||
|
0, // 0: header.HeaderService.Header:input_type -> header.headerRequest
|
||||||
|
1, // 1: header.HeaderService.Header:output_type -> header.headerResponse
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_headers_headers_proto_init() }
|
||||||
|
func file_headers_headers_proto_init() {
|
||||||
|
if File_headers_headers_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_headers_headers_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*HeaderRequest); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file_headers_headers_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*HeaderResponse); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_headers_headers_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 2,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_headers_headers_proto_goTypes,
|
||||||
|
DependencyIndexes: file_headers_headers_proto_depIdxs,
|
||||||
|
MessageInfos: file_headers_headers_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_headers_headers_proto = out.File
|
||||||
|
file_headers_headers_proto_rawDesc = nil
|
||||||
|
file_headers_headers_proto_goTypes = nil
|
||||||
|
file_headers_headers_proto_depIdxs = nil
|
||||||
|
}
|
||||||
15
pkg/api/grpc/headers/headers.proto
Normal file
15
pkg/api/grpc/headers/headers.proto
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./header";
|
||||||
|
|
||||||
|
package header;
|
||||||
|
|
||||||
|
message HeaderRequest {}
|
||||||
|
|
||||||
|
message HeaderResponse {
|
||||||
|
repeated string headers = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
service HeaderService {
|
||||||
|
rpc Header(HeaderRequest) returns (HeaderResponse) {}
|
||||||
|
}
|
||||||
105
pkg/api/grpc/headers/headers_grpc.pb.go
Normal file
105
pkg/api/grpc/headers/headers_grpc.pb.go
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.2.0
|
||||||
|
// - protoc v4.25.0
|
||||||
|
// source: headers/headers.proto
|
||||||
|
|
||||||
|
package header
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
// HeaderServiceClient is the client API for HeaderService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type HeaderServiceClient interface {
|
||||||
|
Header(ctx context.Context, in *HeaderRequest, opts ...grpc.CallOption) (*HeaderResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type headerServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHeaderServiceClient(cc grpc.ClientConnInterface) HeaderServiceClient {
|
||||||
|
return &headerServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *headerServiceClient) Header(ctx context.Context, in *HeaderRequest, opts ...grpc.CallOption) (*HeaderResponse, error) {
|
||||||
|
out := new(HeaderResponse)
|
||||||
|
err := c.cc.Invoke(ctx, "/header.HeaderService/Header", in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// HeaderServiceServer is the server API for HeaderService service.
|
||||||
|
// All implementations must embed UnimplementedHeaderServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type HeaderServiceServer interface {
|
||||||
|
Header(context.Context, *HeaderRequest) (*HeaderResponse, error)
|
||||||
|
mustEmbedUnimplementedHeaderServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedHeaderServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedHeaderServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedHeaderServiceServer) Header(context.Context, *HeaderRequest) (*HeaderResponse, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Header not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedHeaderServiceServer) mustEmbedUnimplementedHeaderServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafeHeaderServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to HeaderServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafeHeaderServiceServer interface {
|
||||||
|
mustEmbedUnimplementedHeaderServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterHeaderServiceServer(s grpc.ServiceRegistrar, srv HeaderServiceServer) {
|
||||||
|
s.RegisterService(&HeaderService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _HeaderService_Header_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(HeaderRequest)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(HeaderServiceServer).Header(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: "/header.HeaderService/Header",
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(HeaderServiceServer).Header(ctx, req.(*HeaderRequest))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HeaderService_ServiceDesc is the grpc.ServiceDesc for HeaderService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var HeaderService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "header.HeaderService",
|
||||||
|
HandlerType: (*HeaderServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Header",
|
||||||
|
Handler: _HeaderService_Header_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "headers/headers.proto",
|
||||||
|
}
|
||||||
69
pkg/api/grpc/headers_test.go
Normal file
69
pkg/api/grpc/headers_test.go
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"regexp"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/headers"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/metadata"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGrpcHeader(t *testing.T) {
|
||||||
|
|
||||||
|
lis := bufconn.Listen(1024 * 1024)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
lis.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
t.Cleanup(func() {
|
||||||
|
srv.Stop()
|
||||||
|
})
|
||||||
|
|
||||||
|
header.RegisterHeaderServiceServer(srv, &HeaderServer{})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(lis); err != nil {
|
||||||
|
log.Fatalf("srv.Serve %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
dialer := func(context.Context, string) (net.Conn, error) {
|
||||||
|
return lis.Dial()
|
||||||
|
}
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(context.Background(), "", grpc.WithContextDialer(dialer), grpc.WithInsecure())
|
||||||
|
t.Cleanup(func() {
|
||||||
|
conn.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("grpc.DialContext %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
headers := metadata.New(map[string]string{
|
||||||
|
"X-Test": "testing",
|
||||||
|
})
|
||||||
|
|
||||||
|
ctx := metadata.NewOutgoingContext(context.Background(), headers)
|
||||||
|
|
||||||
|
client := header.NewHeaderServiceClient(conn)
|
||||||
|
res, err := client.Header(ctx, &header.HeaderRequest{})
|
||||||
|
|
||||||
|
if _, ok := status.FromError(err); !ok {
|
||||||
|
t.Errorf("Header returned type %T, want %T", err, status.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := ".*testing.*"
|
||||||
|
r := regexp.MustCompile(expected)
|
||||||
|
if !r.MatchString(res.String()) {
|
||||||
|
t.Fatalf("Returned unexpected body:\ngot \n%v \nwant \n%s",
|
||||||
|
res, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
70
pkg/api/grpc/info.go
Normal file
70
pkg/api/grpc/info.go
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"log"
|
||||||
|
"runtime"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/info"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/version"
|
||||||
|
)
|
||||||
|
|
||||||
|
type infoServer struct {
|
||||||
|
pb.UnimplementedInfoServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *infoServer) Info(ctx context.Context, message *pb.InfoRequest) (*pb.InfoResponse, error) {
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if r := recover(); r != nil {
|
||||||
|
log.Println("Recovered from panic:", r)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
data := RuntimeResponse{
|
||||||
|
Hostname: s.config.Hostname,
|
||||||
|
Version: version.VERSION,
|
||||||
|
Revision: version.REVISION,
|
||||||
|
Color: s.config.UIColor,
|
||||||
|
Logo: s.config.UILogo,
|
||||||
|
Message: s.config.UIMessage,
|
||||||
|
Goos: runtime.GOOS,
|
||||||
|
Goarch: runtime.GOARCH,
|
||||||
|
Runtime: runtime.Version(),
|
||||||
|
Numgoroutine: strconv.FormatInt(int64(runtime.NumGoroutine()), 10),
|
||||||
|
Numcpu: strconv.FormatInt(int64(runtime.NumCPU()), 10),
|
||||||
|
}
|
||||||
|
|
||||||
|
return &pb.InfoResponse{
|
||||||
|
Hostname: data.Hostname,
|
||||||
|
Version: data.Version,
|
||||||
|
Revision: data.Revision,
|
||||||
|
Color: data.Color,
|
||||||
|
Logo: data.Logo,
|
||||||
|
Message: data.Message,
|
||||||
|
Goos: data.Goos,
|
||||||
|
Goarch: data.Goarch,
|
||||||
|
Runtime: data.Runtime,
|
||||||
|
Numgoroutine: data.Numgoroutine,
|
||||||
|
Numcpu: data.Numcpu,
|
||||||
|
}, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
type RuntimeResponse struct {
|
||||||
|
Hostname string `json:"hostname"`
|
||||||
|
Version string `json:"version"`
|
||||||
|
Revision string `json:"revision"`
|
||||||
|
Color string `json:"color"`
|
||||||
|
Logo string `json:"logo"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Goos string `json:"goos"`
|
||||||
|
Goarch string `json:"goarch"`
|
||||||
|
Runtime string `json:"runtime"`
|
||||||
|
Numgoroutine string `json:"num_goroutine"`
|
||||||
|
Numcpu string `json:"num_cpu"`
|
||||||
|
}
|
||||||
296
pkg/api/grpc/info/info.pb.go
Normal file
296
pkg/api/grpc/info/info.pb.go
Normal file
@@ -0,0 +1,296 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.28.1
|
||||||
|
// protoc v4.25.0
|
||||||
|
// source: info/info.proto
|
||||||
|
|
||||||
|
package info
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type InfoRequest struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoRequest) Reset() {
|
||||||
|
*x = InfoRequest{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_info_info_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoRequest) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*InfoRequest) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *InfoRequest) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_info_info_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use InfoRequest.ProtoReflect.Descriptor instead.
|
||||||
|
func (*InfoRequest) Descriptor() ([]byte, []int) {
|
||||||
|
return file_info_info_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
type InfoResponse struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
|
||||||
|
Hostname string `protobuf:"bytes,1,opt,name=hostname,proto3" json:"hostname,omitempty"`
|
||||||
|
Version string `protobuf:"bytes,2,opt,name=version,proto3" json:"version,omitempty"`
|
||||||
|
Revision string `protobuf:"bytes,3,opt,name=revision,proto3" json:"revision,omitempty"`
|
||||||
|
Color string `protobuf:"bytes,4,opt,name=color,proto3" json:"color,omitempty"`
|
||||||
|
Logo string `protobuf:"bytes,5,opt,name=logo,proto3" json:"logo,omitempty"`
|
||||||
|
Message string `protobuf:"bytes,6,opt,name=message,proto3" json:"message,omitempty"`
|
||||||
|
Goos string `protobuf:"bytes,7,opt,name=goos,proto3" json:"goos,omitempty"`
|
||||||
|
Goarch string `protobuf:"bytes,8,opt,name=goarch,proto3" json:"goarch,omitempty"`
|
||||||
|
Runtime string `protobuf:"bytes,9,opt,name=runtime,proto3" json:"runtime,omitempty"`
|
||||||
|
Numgoroutine string `protobuf:"bytes,10,opt,name=numgoroutine,proto3" json:"numgoroutine,omitempty"`
|
||||||
|
Numcpu string `protobuf:"bytes,11,opt,name=numcpu,proto3" json:"numcpu,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) Reset() {
|
||||||
|
*x = InfoResponse{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_info_info_proto_msgTypes[1]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*InfoResponse) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *InfoResponse) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_info_info_proto_msgTypes[1]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use InfoResponse.ProtoReflect.Descriptor instead.
|
||||||
|
func (*InfoResponse) Descriptor() ([]byte, []int) {
|
||||||
|
return file_info_info_proto_rawDescGZIP(), []int{1}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetHostname() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Hostname
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetVersion() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Version
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetRevision() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Revision
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetColor() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Color
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetLogo() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Logo
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetMessage() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Message
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetGoos() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Goos
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetGoarch() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Goarch
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetRuntime() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Runtime
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetNumgoroutine() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Numgoroutine
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *InfoResponse) GetNumcpu() string {
|
||||||
|
if x != nil {
|
||||||
|
return x.Numcpu
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_info_info_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_info_info_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x0f, 0x69, 0x6e, 0x66, 0x6f, 0x2f, 0x69, 0x6e, 0x66, 0x6f, 0x2e, 0x70, 0x72, 0x6f, 0x74,
|
||||||
|
0x6f, 0x12, 0x04, 0x69, 0x6e, 0x66, 0x6f, 0x22, 0x0d, 0x0a, 0x0b, 0x49, 0x6e, 0x66, 0x6f, 0x52,
|
||||||
|
0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x22, 0xa6, 0x02, 0x0a, 0x0c, 0x49, 0x6e, 0x66, 0x6f, 0x52,
|
||||||
|
0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x1a, 0x0a, 0x08, 0x68, 0x6f, 0x73, 0x74, 0x6e,
|
||||||
|
0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x68, 0x6f, 0x73, 0x74, 0x6e,
|
||||||
|
0x61, 0x6d, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x02,
|
||||||
|
0x20, 0x01, 0x28, 0x09, 0x52, 0x07, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x12, 0x1a, 0x0a,
|
||||||
|
0x08, 0x72, 0x65, 0x76, 0x69, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52,
|
||||||
|
0x08, 0x72, 0x65, 0x76, 0x69, 0x73, 0x69, 0x6f, 0x6e, 0x12, 0x14, 0x0a, 0x05, 0x63, 0x6f, 0x6c,
|
||||||
|
0x6f, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x63, 0x6f, 0x6c, 0x6f, 0x72, 0x12,
|
||||||
|
0x12, 0x0a, 0x04, 0x6c, 0x6f, 0x67, 0x6f, 0x18, 0x05, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6c,
|
||||||
|
0x6f, 0x67, 0x6f, 0x12, 0x18, 0x0a, 0x07, 0x6d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x18, 0x06,
|
||||||
|
0x20, 0x01, 0x28, 0x09, 0x52, 0x07, 0x6d, 0x65, 0x73, 0x73, 0x61, 0x67, 0x65, 0x12, 0x12, 0x0a,
|
||||||
|
0x04, 0x67, 0x6f, 0x6f, 0x73, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x67, 0x6f, 0x6f,
|
||||||
|
0x73, 0x12, 0x16, 0x0a, 0x06, 0x67, 0x6f, 0x61, 0x72, 0x63, 0x68, 0x18, 0x08, 0x20, 0x01, 0x28,
|
||||||
|
0x09, 0x52, 0x06, 0x67, 0x6f, 0x61, 0x72, 0x63, 0x68, 0x12, 0x18, 0x0a, 0x07, 0x72, 0x75, 0x6e,
|
||||||
|
0x74, 0x69, 0x6d, 0x65, 0x18, 0x09, 0x20, 0x01, 0x28, 0x09, 0x52, 0x07, 0x72, 0x75, 0x6e, 0x74,
|
||||||
|
0x69, 0x6d, 0x65, 0x12, 0x22, 0x0a, 0x0c, 0x6e, 0x75, 0x6d, 0x67, 0x6f, 0x72, 0x6f, 0x75, 0x74,
|
||||||
|
0x69, 0x6e, 0x65, 0x18, 0x0a, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0c, 0x6e, 0x75, 0x6d, 0x67, 0x6f,
|
||||||
|
0x72, 0x6f, 0x75, 0x74, 0x69, 0x6e, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x6e, 0x75, 0x6d, 0x63, 0x70,
|
||||||
|
0x75, 0x18, 0x0b, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x6e, 0x75, 0x6d, 0x63, 0x70, 0x75, 0x32,
|
||||||
|
0x3e, 0x0a, 0x0b, 0x49, 0x6e, 0x66, 0x6f, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x2f,
|
||||||
|
0x0a, 0x04, 0x49, 0x6e, 0x66, 0x6f, 0x12, 0x11, 0x2e, 0x69, 0x6e, 0x66, 0x6f, 0x2e, 0x49, 0x6e,
|
||||||
|
0x66, 0x6f, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x12, 0x2e, 0x69, 0x6e, 0x66, 0x6f,
|
||||||
|
0x2e, 0x49, 0x6e, 0x66, 0x6f, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42,
|
||||||
|
0x08, 0x5a, 0x06, 0x2e, 0x2f, 0x69, 0x6e, 0x66, 0x6f, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f,
|
||||||
|
0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_info_info_proto_rawDescOnce sync.Once
|
||||||
|
file_info_info_proto_rawDescData = file_info_info_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_info_info_proto_rawDescGZIP() []byte {
|
||||||
|
file_info_info_proto_rawDescOnce.Do(func() {
|
||||||
|
file_info_info_proto_rawDescData = protoimpl.X.CompressGZIP(file_info_info_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_info_info_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_info_info_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
|
||||||
|
var file_info_info_proto_goTypes = []interface{}{
|
||||||
|
(*InfoRequest)(nil), // 0: info.InfoRequest
|
||||||
|
(*InfoResponse)(nil), // 1: info.InfoResponse
|
||||||
|
}
|
||||||
|
var file_info_info_proto_depIdxs = []int32{
|
||||||
|
0, // 0: info.InfoService.Info:input_type -> info.InfoRequest
|
||||||
|
1, // 1: info.InfoService.Info:output_type -> info.InfoResponse
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_info_info_proto_init() }
|
||||||
|
func file_info_info_proto_init() {
|
||||||
|
if File_info_info_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_info_info_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*InfoRequest); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file_info_info_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*InfoResponse); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_info_info_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 2,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_info_info_proto_goTypes,
|
||||||
|
DependencyIndexes: file_info_info_proto_depIdxs,
|
||||||
|
MessageInfos: file_info_info_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_info_info_proto = out.File
|
||||||
|
file_info_info_proto_rawDesc = nil
|
||||||
|
file_info_info_proto_goTypes = nil
|
||||||
|
file_info_info_proto_depIdxs = nil
|
||||||
|
}
|
||||||
25
pkg/api/grpc/info/info.proto
Normal file
25
pkg/api/grpc/info/info.proto
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./info";
|
||||||
|
|
||||||
|
package info;
|
||||||
|
|
||||||
|
message InfoRequest {}
|
||||||
|
|
||||||
|
message InfoResponse {
|
||||||
|
string hostname = 1;
|
||||||
|
string version = 2;
|
||||||
|
string revision = 3;
|
||||||
|
string color = 4 ;
|
||||||
|
string logo = 5 ;
|
||||||
|
string message = 6 ;
|
||||||
|
string goos = 7;
|
||||||
|
string goarch = 8;
|
||||||
|
string runtime = 9;
|
||||||
|
string numgoroutine = 10;
|
||||||
|
string numcpu = 11;
|
||||||
|
}
|
||||||
|
|
||||||
|
service InfoService {
|
||||||
|
rpc Info (InfoRequest) returns (InfoResponse) {}
|
||||||
|
}
|
||||||
105
pkg/api/grpc/info/info_grpc.pb.go
Normal file
105
pkg/api/grpc/info/info_grpc.pb.go
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.2.0
|
||||||
|
// - protoc v4.25.0
|
||||||
|
// source: info/info.proto
|
||||||
|
|
||||||
|
package info
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
// InfoServiceClient is the client API for InfoService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type InfoServiceClient interface {
|
||||||
|
Info(ctx context.Context, in *InfoRequest, opts ...grpc.CallOption) (*InfoResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type infoServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewInfoServiceClient(cc grpc.ClientConnInterface) InfoServiceClient {
|
||||||
|
return &infoServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *infoServiceClient) Info(ctx context.Context, in *InfoRequest, opts ...grpc.CallOption) (*InfoResponse, error) {
|
||||||
|
out := new(InfoResponse)
|
||||||
|
err := c.cc.Invoke(ctx, "/info.InfoService/Info", in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// InfoServiceServer is the server API for InfoService service.
|
||||||
|
// All implementations must embed UnimplementedInfoServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type InfoServiceServer interface {
|
||||||
|
Info(context.Context, *InfoRequest) (*InfoResponse, error)
|
||||||
|
mustEmbedUnimplementedInfoServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedInfoServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedInfoServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedInfoServiceServer) Info(context.Context, *InfoRequest) (*InfoResponse, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Info not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedInfoServiceServer) mustEmbedUnimplementedInfoServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafeInfoServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to InfoServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafeInfoServiceServer interface {
|
||||||
|
mustEmbedUnimplementedInfoServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterInfoServiceServer(s grpc.ServiceRegistrar, srv InfoServiceServer) {
|
||||||
|
s.RegisterService(&InfoService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _InfoService_Info_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(InfoRequest)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(InfoServiceServer).Info(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: "/info.InfoService/Info",
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(InfoServiceServer).Info(ctx, req.(*InfoRequest))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InfoService_ServiceDesc is the grpc.ServiceDesc for InfoService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var InfoService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "info.InfoService",
|
||||||
|
HandlerType: (*InfoServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Info",
|
||||||
|
Handler: _InfoService_Info_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "info/info.proto",
|
||||||
|
}
|
||||||
65
pkg/api/grpc/info_test.go
Normal file
65
pkg/api/grpc/info_test.go
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"regexp"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/info"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGrpcInfo(t *testing.T) {
|
||||||
|
|
||||||
|
lis := bufconn.Listen(1024 * 1024)
|
||||||
|
t.Cleanup(func() {
|
||||||
|
lis.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
s := NewMockGrpcServer()
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
t.Cleanup(func() {
|
||||||
|
srv.Stop()
|
||||||
|
})
|
||||||
|
|
||||||
|
info.RegisterInfoServiceServer(srv, &infoServer{config: s.config})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(lis); err != nil {
|
||||||
|
log.Fatalf("srv.Serve %v", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
dialer := func(context.Context, string) (net.Conn, error) {
|
||||||
|
return lis.Dial()
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
conn, err := grpc.DialContext(ctx, "", grpc.WithContextDialer(dialer), grpc.WithInsecure())
|
||||||
|
t.Cleanup(func() {
|
||||||
|
conn.Close()
|
||||||
|
})
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("grpc.DialContext %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := info.NewInfoServiceClient(conn)
|
||||||
|
res, err := client.Info(context.Background(), &info.InfoRequest{})
|
||||||
|
|
||||||
|
if _, ok := status.FromError(err); !ok {
|
||||||
|
t.Errorf("Info returned type %T, want %T", err, status.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := ".*color.*blue.*"
|
||||||
|
r := regexp.MustCompile(expected)
|
||||||
|
if !r.MatchString(res.String()) {
|
||||||
|
t.Fatalf("Returned unexpected body:\ngot \n%v \nwant \n%s",
|
||||||
|
res.Color, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
30
pkg/api/grpc/mock_grpc.go
Normal file
30
pkg/api/grpc/mock_grpc.go
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NewMockGrpcServer() *Server {
|
||||||
|
config := &Config{
|
||||||
|
Port: 9999,
|
||||||
|
// ServerShutdownTimeout: 5 * time.Second,
|
||||||
|
// HttpServerTimeout: 30 * time.Second,
|
||||||
|
BackendURL: []string{},
|
||||||
|
ConfigPath: "/config",
|
||||||
|
DataPath: "/data",
|
||||||
|
// HttpClientTimeout: 30 * time.Second,
|
||||||
|
UIColor: "blue",
|
||||||
|
UIPath: ".ui",
|
||||||
|
UIMessage: "Greetings",
|
||||||
|
Hostname: "localhost",
|
||||||
|
}
|
||||||
|
|
||||||
|
logger, _ := zap.NewDevelopment()
|
||||||
|
|
||||||
|
return &Server{
|
||||||
|
//router: mux.NewRouter(),
|
||||||
|
logger: logger,
|
||||||
|
config: config,
|
||||||
|
//tracer: trace.NewNoopTracerProvider().Tracer("mock"),
|
||||||
|
}
|
||||||
|
}
|
||||||
22
pkg/api/grpc/panic.go
Normal file
22
pkg/api/grpc/panic.go
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
// "log"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/panic"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
)
|
||||||
|
|
||||||
|
type PanicServer struct {
|
||||||
|
pb.UnimplementedPanicServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PanicServer) Panic(ctx context.Context, req *pb.PanicRequest) (*pb.PanicResponse, error) {
|
||||||
|
s.logger.Info("Panic command received")
|
||||||
|
os.Exit(225)
|
||||||
|
return &pb.PanicResponse{}, nil
|
||||||
|
}
|
||||||
189
pkg/api/grpc/panic/panic.pb.go
Normal file
189
pkg/api/grpc/panic/panic.pb.go
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
// Code generated by protoc-gen-go. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// protoc-gen-go v1.28.1
|
||||||
|
// protoc v3.6.1
|
||||||
|
// source: panic.proto
|
||||||
|
|
||||||
|
package panic
|
||||||
|
|
||||||
|
import (
|
||||||
|
protoreflect "google.golang.org/protobuf/reflect/protoreflect"
|
||||||
|
protoimpl "google.golang.org/protobuf/runtime/protoimpl"
|
||||||
|
reflect "reflect"
|
||||||
|
sync "sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Verify that this generated code is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion)
|
||||||
|
// Verify that runtime/protoimpl is sufficiently up-to-date.
|
||||||
|
_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20)
|
||||||
|
)
|
||||||
|
|
||||||
|
type PanicRequest struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PanicRequest) Reset() {
|
||||||
|
*x = PanicRequest{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_panic_proto_msgTypes[0]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PanicRequest) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*PanicRequest) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *PanicRequest) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_panic_proto_msgTypes[0]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use PanicRequest.ProtoReflect.Descriptor instead.
|
||||||
|
func (*PanicRequest) Descriptor() ([]byte, []int) {
|
||||||
|
return file_panic_proto_rawDescGZIP(), []int{0}
|
||||||
|
}
|
||||||
|
|
||||||
|
type PanicResponse struct {
|
||||||
|
state protoimpl.MessageState
|
||||||
|
sizeCache protoimpl.SizeCache
|
||||||
|
unknownFields protoimpl.UnknownFields
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PanicResponse) Reset() {
|
||||||
|
*x = PanicResponse{}
|
||||||
|
if protoimpl.UnsafeEnabled {
|
||||||
|
mi := &file_panic_proto_msgTypes[1]
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x *PanicResponse) String() string {
|
||||||
|
return protoimpl.X.MessageStringOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*PanicResponse) ProtoMessage() {}
|
||||||
|
|
||||||
|
func (x *PanicResponse) ProtoReflect() protoreflect.Message {
|
||||||
|
mi := &file_panic_proto_msgTypes[1]
|
||||||
|
if protoimpl.UnsafeEnabled && x != nil {
|
||||||
|
ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
|
||||||
|
if ms.LoadMessageInfo() == nil {
|
||||||
|
ms.StoreMessageInfo(mi)
|
||||||
|
}
|
||||||
|
return ms
|
||||||
|
}
|
||||||
|
return mi.MessageOf(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated: Use PanicResponse.ProtoReflect.Descriptor instead.
|
||||||
|
func (*PanicResponse) Descriptor() ([]byte, []int) {
|
||||||
|
return file_panic_proto_rawDescGZIP(), []int{1}
|
||||||
|
}
|
||||||
|
|
||||||
|
var File_panic_proto protoreflect.FileDescriptor
|
||||||
|
|
||||||
|
var file_panic_proto_rawDesc = []byte{
|
||||||
|
0x0a, 0x0b, 0x70, 0x61, 0x6e, 0x69, 0x63, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x05, 0x70,
|
||||||
|
0x61, 0x6e, 0x69, 0x63, 0x22, 0x0e, 0x0a, 0x0c, 0x50, 0x61, 0x6e, 0x69, 0x63, 0x52, 0x65, 0x71,
|
||||||
|
0x75, 0x65, 0x73, 0x74, 0x22, 0x0f, 0x0a, 0x0d, 0x50, 0x61, 0x6e, 0x69, 0x63, 0x52, 0x65, 0x73,
|
||||||
|
0x70, 0x6f, 0x6e, 0x73, 0x65, 0x32, 0x44, 0x0a, 0x0c, 0x50, 0x61, 0x6e, 0x69, 0x63, 0x53, 0x65,
|
||||||
|
0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x34, 0x0a, 0x05, 0x50, 0x61, 0x6e, 0x69, 0x63, 0x12, 0x13,
|
||||||
|
0x2e, 0x70, 0x61, 0x6e, 0x69, 0x63, 0x2e, 0x50, 0x61, 0x6e, 0x69, 0x63, 0x52, 0x65, 0x71, 0x75,
|
||||||
|
0x65, 0x73, 0x74, 0x1a, 0x14, 0x2e, 0x70, 0x61, 0x6e, 0x69, 0x63, 0x2e, 0x50, 0x61, 0x6e, 0x69,
|
||||||
|
0x63, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0x09, 0x5a, 0x07, 0x2e,
|
||||||
|
0x2f, 0x70, 0x61, 0x6e, 0x69, 0x63, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
file_panic_proto_rawDescOnce sync.Once
|
||||||
|
file_panic_proto_rawDescData = file_panic_proto_rawDesc
|
||||||
|
)
|
||||||
|
|
||||||
|
func file_panic_proto_rawDescGZIP() []byte {
|
||||||
|
file_panic_proto_rawDescOnce.Do(func() {
|
||||||
|
file_panic_proto_rawDescData = protoimpl.X.CompressGZIP(file_panic_proto_rawDescData)
|
||||||
|
})
|
||||||
|
return file_panic_proto_rawDescData
|
||||||
|
}
|
||||||
|
|
||||||
|
var file_panic_proto_msgTypes = make([]protoimpl.MessageInfo, 2)
|
||||||
|
var file_panic_proto_goTypes = []interface{}{
|
||||||
|
(*PanicRequest)(nil), // 0: panic.PanicRequest
|
||||||
|
(*PanicResponse)(nil), // 1: panic.PanicResponse
|
||||||
|
}
|
||||||
|
var file_panic_proto_depIdxs = []int32{
|
||||||
|
0, // 0: panic.PanicService.Panic:input_type -> panic.PanicRequest
|
||||||
|
1, // 1: panic.PanicService.Panic:output_type -> panic.PanicResponse
|
||||||
|
1, // [1:2] is the sub-list for method output_type
|
||||||
|
0, // [0:1] is the sub-list for method input_type
|
||||||
|
0, // [0:0] is the sub-list for extension type_name
|
||||||
|
0, // [0:0] is the sub-list for extension extendee
|
||||||
|
0, // [0:0] is the sub-list for field type_name
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() { file_panic_proto_init() }
|
||||||
|
func file_panic_proto_init() {
|
||||||
|
if File_panic_proto != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !protoimpl.UnsafeEnabled {
|
||||||
|
file_panic_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*PanicRequest); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file_panic_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} {
|
||||||
|
switch v := v.(*PanicResponse); i {
|
||||||
|
case 0:
|
||||||
|
return &v.state
|
||||||
|
case 1:
|
||||||
|
return &v.sizeCache
|
||||||
|
case 2:
|
||||||
|
return &v.unknownFields
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
type x struct{}
|
||||||
|
out := protoimpl.TypeBuilder{
|
||||||
|
File: protoimpl.DescBuilder{
|
||||||
|
GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
|
||||||
|
RawDescriptor: file_panic_proto_rawDesc,
|
||||||
|
NumEnums: 0,
|
||||||
|
NumMessages: 2,
|
||||||
|
NumExtensions: 0,
|
||||||
|
NumServices: 1,
|
||||||
|
},
|
||||||
|
GoTypes: file_panic_proto_goTypes,
|
||||||
|
DependencyIndexes: file_panic_proto_depIdxs,
|
||||||
|
MessageInfos: file_panic_proto_msgTypes,
|
||||||
|
}.Build()
|
||||||
|
File_panic_proto = out.File
|
||||||
|
file_panic_proto_rawDesc = nil
|
||||||
|
file_panic_proto_goTypes = nil
|
||||||
|
file_panic_proto_depIdxs = nil
|
||||||
|
}
|
||||||
17
pkg/api/grpc/panic/panic.proto
Normal file
17
pkg/api/grpc/panic/panic.proto
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "./panic";
|
||||||
|
|
||||||
|
package panic;
|
||||||
|
|
||||||
|
// The greeting service definition.
|
||||||
|
service PanicService {
|
||||||
|
rpc Panic (PanicRequest) returns (PanicResponse) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
message PanicRequest {
|
||||||
|
}
|
||||||
|
|
||||||
|
message PanicResponse {
|
||||||
|
}
|
||||||
|
|
||||||
105
pkg/api/grpc/panic/panic_grpc.pb.go
Normal file
105
pkg/api/grpc/panic/panic_grpc.pb.go
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Code generated by protoc-gen-go-grpc. DO NOT EDIT.
|
||||||
|
// versions:
|
||||||
|
// - protoc-gen-go-grpc v1.2.0
|
||||||
|
// - protoc v3.6.1
|
||||||
|
// source: panic.proto
|
||||||
|
|
||||||
|
package panic
|
||||||
|
|
||||||
|
import (
|
||||||
|
context "context"
|
||||||
|
grpc "google.golang.org/grpc"
|
||||||
|
codes "google.golang.org/grpc/codes"
|
||||||
|
status "google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a compile-time assertion to ensure that this generated file
|
||||||
|
// is compatible with the grpc package it is being compiled against.
|
||||||
|
// Requires gRPC-Go v1.32.0 or later.
|
||||||
|
const _ = grpc.SupportPackageIsVersion7
|
||||||
|
|
||||||
|
// PanicServiceClient is the client API for PanicService service.
|
||||||
|
//
|
||||||
|
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream.
|
||||||
|
type PanicServiceClient interface {
|
||||||
|
Panic(ctx context.Context, in *PanicRequest, opts ...grpc.CallOption) (*PanicResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type panicServiceClient struct {
|
||||||
|
cc grpc.ClientConnInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewPanicServiceClient(cc grpc.ClientConnInterface) PanicServiceClient {
|
||||||
|
return &panicServiceClient{cc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *panicServiceClient) Panic(ctx context.Context, in *PanicRequest, opts ...grpc.CallOption) (*PanicResponse, error) {
|
||||||
|
out := new(PanicResponse)
|
||||||
|
err := c.cc.Invoke(ctx, "/panic.PanicService/Panic", in, out, opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PanicServiceServer is the server API for PanicService service.
|
||||||
|
// All implementations must embed UnimplementedPanicServiceServer
|
||||||
|
// for forward compatibility
|
||||||
|
type PanicServiceServer interface {
|
||||||
|
Panic(context.Context, *PanicRequest) (*PanicResponse, error)
|
||||||
|
mustEmbedUnimplementedPanicServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnimplementedPanicServiceServer must be embedded to have forward compatible implementations.
|
||||||
|
type UnimplementedPanicServiceServer struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UnimplementedPanicServiceServer) Panic(context.Context, *PanicRequest) (*PanicResponse, error) {
|
||||||
|
return nil, status.Errorf(codes.Unimplemented, "method Panic not implemented")
|
||||||
|
}
|
||||||
|
func (UnimplementedPanicServiceServer) mustEmbedUnimplementedPanicServiceServer() {}
|
||||||
|
|
||||||
|
// UnsafePanicServiceServer may be embedded to opt out of forward compatibility for this service.
|
||||||
|
// Use of this interface is not recommended, as added methods to PanicServiceServer will
|
||||||
|
// result in compilation errors.
|
||||||
|
type UnsafePanicServiceServer interface {
|
||||||
|
mustEmbedUnimplementedPanicServiceServer()
|
||||||
|
}
|
||||||
|
|
||||||
|
func RegisterPanicServiceServer(s grpc.ServiceRegistrar, srv PanicServiceServer) {
|
||||||
|
s.RegisterService(&PanicService_ServiceDesc, srv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _PanicService_Panic_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
|
||||||
|
in := new(PanicRequest)
|
||||||
|
if err := dec(in); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if interceptor == nil {
|
||||||
|
return srv.(PanicServiceServer).Panic(ctx, in)
|
||||||
|
}
|
||||||
|
info := &grpc.UnaryServerInfo{
|
||||||
|
Server: srv,
|
||||||
|
FullMethod: "/panic.PanicService/Panic",
|
||||||
|
}
|
||||||
|
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
|
||||||
|
return srv.(PanicServiceServer).Panic(ctx, req.(*PanicRequest))
|
||||||
|
}
|
||||||
|
return interceptor(ctx, in, info, handler)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PanicService_ServiceDesc is the grpc.ServiceDesc for PanicService service.
|
||||||
|
// It's only intended for direct use with grpc.RegisterService,
|
||||||
|
// and not to be introspected or modified (even as a copy)
|
||||||
|
var PanicService_ServiceDesc = grpc.ServiceDesc{
|
||||||
|
ServiceName: "panic.PanicService",
|
||||||
|
HandlerType: (*PanicServiceServer)(nil),
|
||||||
|
Methods: []grpc.MethodDesc{
|
||||||
|
{
|
||||||
|
MethodName: "Panic",
|
||||||
|
Handler: _PanicService_Panic_Handler,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Streams: []grpc.StreamDesc{},
|
||||||
|
Metadata: "panic.proto",
|
||||||
|
}
|
||||||
99
pkg/api/grpc/server.go
Normal file
99
pkg/api/grpc/server.go
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/echo"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/health"
|
||||||
|
"google.golang.org/grpc/health/grpc_health_v1"
|
||||||
|
"google.golang.org/grpc/reflection"
|
||||||
|
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/delay"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/env"
|
||||||
|
header "github.com/stefanprodan/podinfo/pkg/api/grpc/headers"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/info"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/panic"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/status"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/token"
|
||||||
|
"github.com/stefanprodan/podinfo/pkg/api/grpc/version"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Server struct {
|
||||||
|
logger *zap.Logger
|
||||||
|
config *Config
|
||||||
|
}
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
Port int `mapstructure:"grpc-port"`
|
||||||
|
ServiceName string `mapstructure:"grpc-service-name"`
|
||||||
|
|
||||||
|
BackendURL []string `mapstructure:"backend-url"`
|
||||||
|
UILogo string `mapstructure:"ui-logo"`
|
||||||
|
UIMessage string `mapstructure:"ui-message"`
|
||||||
|
UIColor string `mapstructure:"ui-color"`
|
||||||
|
UIPath string `mapstructure:"ui-path"`
|
||||||
|
DataPath string `mapstructure:"data-path"`
|
||||||
|
ConfigPath string `mapstructure:"config-path"`
|
||||||
|
CertPath string `mapstructure:"cert-path"`
|
||||||
|
Host string `mapstructure:"host"`
|
||||||
|
//Port string `mapstructure:"port"`
|
||||||
|
SecurePort string `mapstructure:"secure-port"`
|
||||||
|
PortMetrics int `mapstructure:"port-metrics"`
|
||||||
|
Hostname string `mapstructure:"hostname"`
|
||||||
|
H2C bool `mapstructure:"h2c"`
|
||||||
|
RandomDelay bool `mapstructure:"random-delay"`
|
||||||
|
RandomDelayUnit string `mapstructure:"random-delay-unit"`
|
||||||
|
RandomDelayMin int `mapstructure:"random-delay-min"`
|
||||||
|
RandomDelayMax int `mapstructure:"random-delay-max"`
|
||||||
|
RandomError bool `mapstructure:"random-error"`
|
||||||
|
Unhealthy bool `mapstructure:"unhealthy"`
|
||||||
|
Unready bool `mapstructure:"unready"`
|
||||||
|
JWTSecret string `mapstructure:"jwt-secret"`
|
||||||
|
CacheServer string `mapstructure:"cache-server"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewServer(config *Config, logger *zap.Logger) (*Server, error) {
|
||||||
|
srv := &Server{
|
||||||
|
logger: logger,
|
||||||
|
config: config,
|
||||||
|
}
|
||||||
|
|
||||||
|
return srv, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) ListenAndServe() *grpc.Server {
|
||||||
|
listener, err := net.Listen("tcp", fmt.Sprintf(":%v", s.config.Port))
|
||||||
|
if err != nil {
|
||||||
|
s.logger.Fatal("failed to listen", zap.Int("port", s.config.Port))
|
||||||
|
}
|
||||||
|
|
||||||
|
srv := grpc.NewServer()
|
||||||
|
server := health.NewServer()
|
||||||
|
|
||||||
|
// Register grpc apis for reflection
|
||||||
|
echo.RegisterEchoServiceServer(srv, &echoServer{config: s.config, logger: s.logger})
|
||||||
|
version.RegisterVersionServiceServer(srv, &VersionServer{config: s.config, logger: s.logger})
|
||||||
|
panic.RegisterPanicServiceServer(srv, &PanicServer{config: s.config, logger: s.logger})
|
||||||
|
delay.RegisterDelayServiceServer(srv, &DelayServer{config: s.config, logger: s.logger})
|
||||||
|
header.RegisterHeaderServiceServer(srv, &HeaderServer{config: s.config, logger: s.logger})
|
||||||
|
info.RegisterInfoServiceServer(srv, &infoServer{config: s.config})
|
||||||
|
status.RegisterStatusServiceServer(srv, &StatusServer{config: s.config, logger: s.logger})
|
||||||
|
token.RegisterTokenServiceServer(srv, &TokenServer{config: s.config, logger: s.logger})
|
||||||
|
env.RegisterEnvServiceServer(srv, &EnvServer{config: s.config, logger: s.logger})
|
||||||
|
|
||||||
|
reflection.Register(srv)
|
||||||
|
grpc_health_v1.RegisterHealthServer(srv, server)
|
||||||
|
server.SetServingStatus(s.config.ServiceName, grpc_health_v1.HealthCheckResponse_SERVING)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if err := srv.Serve(listener); err != nil {
|
||||||
|
s.logger.Fatal("failed to serve", zap.Error(err))
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
return srv
|
||||||
|
}
|
||||||
47
pkg/api/grpc/status.go
Normal file
47
pkg/api/grpc/status.go
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
package grpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
pb "github.com/stefanprodan/podinfo/pkg/api/grpc/status"
|
||||||
|
"go.uber.org/zap"
|
||||||
|
"google.golang.org/grpc/codes"
|
||||||
|
"google.golang.org/grpc/status"
|
||||||
|
)
|
||||||
|
|
||||||
|
type StatusServer struct {
|
||||||
|
pb.UnimplementedStatusServiceServer
|
||||||
|
config *Config
|
||||||
|
logger *zap.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StatusServer) Status(ctx context.Context, req *pb.StatusRequest) (*pb.StatusResponse, error) {
|
||||||
|
reqCode := req.GetCode()
|
||||||
|
|
||||||
|
grpcCodes := map[string]codes.Code{
|
||||||
|
"Ok": codes.OK,
|
||||||
|
"Canceled": codes.Canceled,
|
||||||
|
"Unknown": codes.Unknown,
|
||||||
|
"InvalidArgument": codes.InvalidArgument,
|
||||||
|
"DeadlineExceeded": codes.DeadlineExceeded,
|
||||||
|
"NotFound": codes.NotFound,
|
||||||
|
"AlreadyExists": codes.AlreadyExists,
|
||||||
|
"PermissionDenied": codes.PermissionDenied,
|
||||||
|
"ResourceExhausted": codes.ResourceExhausted,
|
||||||
|
"FailedPrecondition": codes.FailedPrecondition,
|
||||||
|
"Aborted": codes.Aborted,
|
||||||
|
"OutOfRange": codes.OutOfRange,
|
||||||
|
"Unimplemented": codes.Unimplemented,
|
||||||
|
"Internal": codes.Internal,
|
||||||
|
"Unavailable": codes.Unavailable,
|
||||||
|
"DataLoss": codes.DataLoss,
|
||||||
|
"Unauthenticated": codes.Unauthenticated,
|
||||||
|
}
|
||||||
|
|
||||||
|
code, ok := grpcCodes[reqCode]
|
||||||
|
if !ok {
|
||||||
|
return nil, status.Error(codes.Unknown, "Unknown status code for more information check https://chromium.googlesource.com/external/github.com/grpc/grpc/+/refs/tags/v1.21.4-pre1/doc/statuscodes.md")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &pb.StatusResponse{Status: reqCode}, status.Error(code, "")
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user