From 5349d4831030c30e29775be6ec16179b97fce75e Mon Sep 17 00:00:00 2001 From: Timofey Myagkikh Date: Sun, 19 Nov 2023 00:56:43 +0300 Subject: [PATCH] helm charts --- .github/workflows/{rust.yml => master.yml} | 6 +- build_images.sh | 3 - deploy_helm.sh | 3 + helm/.helmignore | 23 +++ helm/Chart.yaml | 6 + helm/charts/ingress-nginx-4.8.3.tgz | Bin 0 -> 47117 bytes helm/templates/_helpers.tpl | 62 ++++++ helm/templates/acl/deployment.yml | 30 +++ helm/templates/api/deployment.yml | 32 +++ helm/templates/api/ingress.yml | 23 +++ helm/templates/api/service.yml | 10 + helm/templates/kafka/deployment.yml | 33 +++ helm/templates/kafka/service.yml | 10 + helm/templates/mongodb/deployment.yml | 24 +++ helm/templates/mongodb/service.yml | 10 + helm/templates/zookeeper/deployment.yml | 23 +++ helm/templates/zookeeper/service.yml | 10 + helm/values.yaml | 80 ++++++++ .dockerignore => src/.dockerignore | 0 Cargo.lock => src/Cargo.lock | 192 +++++++++--------- Cargo.toml => src/Cargo.toml | 2 +- {acl => src/acl}/Cargo.toml | 0 {acl => src/acl}/Dockerfile | 3 + {acl => src/acl}/src/lib.rs | 0 {acl => src/acl}/src/main.rs | 0 analyze.sh => src/analyze.sh | 0 {app => src/api}/Cargo.toml | 2 +- {app => src/api}/src/endpoints.rs | 0 {app => src/api}/src/lib.rs | 0 {app => src/api}/src/models.rs | 0 build.sh => src/build.sh | 0 src/build_images.sh | 3 + docker-compose.yml => src/docker-compose.yml | 24 ++- {domain => src/domain}/Cargo.toml | 0 {domain => src/domain}/src/commands.rs | 0 {domain => src/domain}/src/events.rs | 0 {domain => src/domain}/src/lib.rs | 0 {domain => src/domain}/src/queries.rs | 0 {domain_impl => src/domain_impl}/Cargo.toml | 0 .../domain_impl}/src/handlers.rs | 0 {domain_impl => src/domain_impl}/src/lib.rs | 0 {domain_impl => src/domain_impl}/src/ports.rs | 0 format.sh => src/format.sh | 0 {host => src/host}/Cargo.toml | 2 +- {host => src/host}/Dockerfile | 7 +- {host => src/host}/src/composition.rs | 10 +- {host => src/host}/src/conf.rs | 0 {host => src/host}/src/lib.rs | 0 {host => src/host}/src/main.rs | 0 {infra => src/infra}/Cargo.toml | 0 {infra => src/infra}/src/lib.rs | 0 {infra => src/infra}/src/repositories.rs | 0 {infra => src/infra}/src/tracker.rs | 0 .../integration-tests}/Cargo.toml | 2 +- .../integration-tests}/tests/gen.rs | 2 +- .../integration-tests}/tests/sut.rs | 8 +- .../integration-tests}/tests/user_tests.rs | 2 +- {messaging => src/messaging}/Cargo.toml | 0 .../messaging}/src/fake_kafka.rs | 0 {messaging => src/messaging}/src/kafka.rs | 0 {messaging => src/messaging}/src/lib.rs | 0 run.sh => src/run.sh | 0 run_tests.sh => src/run_tests.sh | 0 63 files changed, 518 insertions(+), 129 deletions(-) rename .github/workflows/{rust.yml => master.yml} (88%) delete mode 100755 build_images.sh create mode 100755 deploy_helm.sh create mode 100644 helm/.helmignore create mode 100644 helm/Chart.yaml create mode 100644 helm/charts/ingress-nginx-4.8.3.tgz create mode 100644 helm/templates/_helpers.tpl create mode 100644 helm/templates/acl/deployment.yml create mode 100644 helm/templates/api/deployment.yml create mode 100644 helm/templates/api/ingress.yml create mode 100644 helm/templates/api/service.yml create mode 100644 helm/templates/kafka/deployment.yml create mode 100644 helm/templates/kafka/service.yml create mode 100644 helm/templates/mongodb/deployment.yml create mode 100644 helm/templates/mongodb/service.yml create mode 100644 helm/templates/zookeeper/deployment.yml create mode 100644 helm/templates/zookeeper/service.yml create mode 100644 helm/values.yaml rename .dockerignore => src/.dockerignore (100%) rename Cargo.lock => src/Cargo.lock (94%) rename Cargo.toml => src/Cargo.toml (94%) rename {acl => src/acl}/Cargo.toml (100%) rename {acl => src/acl}/Dockerfile (83%) rename {acl => src/acl}/src/lib.rs (100%) rename {acl => src/acl}/src/main.rs (100%) rename analyze.sh => src/analyze.sh (100%) rename {app => src/api}/Cargo.toml (95%) rename {app => src/api}/src/endpoints.rs (100%) rename {app => src/api}/src/lib.rs (100%) rename {app => src/api}/src/models.rs (100%) rename build.sh => src/build.sh (100%) create mode 100755 src/build_images.sh rename docker-compose.yml => src/docker-compose.yml (93%) rename {domain => src/domain}/Cargo.toml (100%) rename {domain => src/domain}/src/commands.rs (100%) rename {domain => src/domain}/src/events.rs (100%) rename {domain => src/domain}/src/lib.rs (100%) rename {domain => src/domain}/src/queries.rs (100%) rename {domain_impl => src/domain_impl}/Cargo.toml (100%) rename {domain_impl => src/domain_impl}/src/handlers.rs (100%) rename {domain_impl => src/domain_impl}/src/lib.rs (100%) rename {domain_impl => src/domain_impl}/src/ports.rs (100%) rename format.sh => src/format.sh (100%) rename {host => src/host}/Cargo.toml (94%) rename {host => src/host}/Dockerfile (74%) rename {host => src/host}/src/composition.rs (94%) rename {host => src/host}/src/conf.rs (100%) rename {host => src/host}/src/lib.rs (100%) rename {host => src/host}/src/main.rs (100%) rename {infra => src/infra}/Cargo.toml (100%) rename {infra => src/infra}/src/lib.rs (100%) rename {infra => src/infra}/src/repositories.rs (100%) rename {infra => src/infra}/src/tracker.rs (100%) rename {integration-tests => src/integration-tests}/Cargo.toml (95%) rename {integration-tests => src/integration-tests}/tests/gen.rs (89%) rename {integration-tests => src/integration-tests}/tests/sut.rs (97%) rename {integration-tests => src/integration-tests}/tests/user_tests.rs (97%) rename {messaging => src/messaging}/Cargo.toml (100%) rename {messaging => src/messaging}/src/fake_kafka.rs (100%) rename {messaging => src/messaging}/src/kafka.rs (100%) rename {messaging => src/messaging}/src/lib.rs (100%) rename run.sh => src/run.sh (100%) rename run_tests.sh => src/run_tests.sh (100%) diff --git a/.github/workflows/rust.yml b/.github/workflows/master.yml similarity index 88% rename from .github/workflows/rust.yml rename to .github/workflows/master.yml index b30f44e..bc9aceb 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/master.yml @@ -11,15 +11,15 @@ env: jobs: build: - runs-on: ubuntu-latest - + defaults: + run: + working-directory: ./src services: mongodb: image: mongo ports: - 27017:27017 - steps: - uses: actions/checkout@v3 - name: Build diff --git a/build_images.sh b/build_images.sh deleted file mode 100755 index 169a51d..0000000 --- a/build_images.sh +++ /dev/null @@ -1,3 +0,0 @@ -#!/bin/bash - -docker-compose build \ No newline at end of file diff --git a/deploy_helm.sh b/deploy_helm.sh new file mode 100755 index 0000000..dab2841 --- /dev/null +++ b/deploy_helm.sh @@ -0,0 +1,3 @@ +#!/bin/bash + +helm install web-api-dev helm \ No newline at end of file diff --git a/helm/.helmignore b/helm/.helmignore new file mode 100644 index 0000000..691fa13 --- /dev/null +++ b/helm/.helmignore @@ -0,0 +1,23 @@ +# Patterns to ignore when building packages. +# This supports shell glob matching, relative path matching, and +# negation (prefixed with !). Only one pattern per line. +.DS_Store +# Common VCS dirs +.git/ +.gitignore +.bzr/ +.bzrignore +.hg/ +.hgignore +.svn/ +# Common backup files +*.swp +*.bak +*.tmp +*.orig +*~ +# Various IDEs +.project +.idea/ +*.tmproj +.vscode/ \ No newline at end of file diff --git a/helm/Chart.yaml b/helm/Chart.yaml new file mode 100644 index 0000000..221d6ea --- /dev/null +++ b/helm/Chart.yaml @@ -0,0 +1,6 @@ +apiVersion: v2 +name: helm +description: Development environment +type: application +version: 0.1.0 +appVersion: "1.16.0" diff --git a/helm/charts/ingress-nginx-4.8.3.tgz b/helm/charts/ingress-nginx-4.8.3.tgz new file mode 100644 index 0000000000000000000000000000000000000000..f2e3f3a08180fdbdd81d3aefcdfc8bca96625883 GIT binary patch literal 47117 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 G(FrnY}Kzl!aAdyQ`;ZY40 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 P0)R$S(O9TeSg8L6uTc1M literal 0 HcmV?d00001 diff --git a/helm/templates/_helpers.tpl b/helm/templates/_helpers.tpl new file mode 100644 index 0000000..ba04c30 --- /dev/null +++ b/helm/templates/_helpers.tpl @@ -0,0 +1,62 @@ +{{/* +Expand the name of the chart. +*/}} +{{- define "helm.name" -}} +{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Create a default fully qualified app name. +We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec). +If release name contains chart name it will be used as a full name. +*/}} +{{- define "helm.fullname" -}} +{{- if .Values.fullnameOverride }} +{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- $name := default .Chart.Name .Values.nameOverride }} +{{- if contains $name .Release.Name }} +{{- .Release.Name | trunc 63 | trimSuffix "-" }} +{{- else }} +{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }} +{{- end }} +{{- end }} +{{- end }} + +{{/* +Create chart name and version as used by the chart label. +*/}} +{{- define "helm.chart" -}} +{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }} +{{- end }} + +{{/* +Common labels +*/}} +{{- define "helm.labels" -}} +helm.sh/chart: {{ include "helm.chart" . }} +{{ include "helm.selectorLabels" . }} +{{- if .Chart.AppVersion }} +app.kubernetes.io/version: {{ .Chart.AppVersion | quote }} +{{- end }} +app.kubernetes.io/managed-by: {{ .Release.Service }} +{{- end }} + +{{/* +Selector labels +*/}} +{{- define "helm.selectorLabels" -}} +app.kubernetes.io/name: {{ include "helm.name" . }} +app.kubernetes.io/instance: {{ .Release.Name }} +{{- end }} + +{{/* +Create the name of the service account to use +*/}} +{{- define "helm.serviceAccountName" -}} +{{- if .Values.serviceAccount.create }} +{{- default (include "helm.fullname" .) .Values.serviceAccount.name }} +{{- else }} +{{- default "default" .Values.serviceAccount.name }} +{{- end }} +{{- end }} diff --git a/helm/templates/acl/deployment.yml b/helm/templates/acl/deployment.yml new file mode 100644 index 0000000..717b97c --- /dev/null +++ b/helm/templates/acl/deployment.yml @@ -0,0 +1,30 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Values.acl.label }}-deployment +spec: + replicas: {{ .Values.acl.replicas }} + selector: + matchLabels: + app: {{ .Values.acl.label }} + template: + metadata: + labels: + app: {{ .Values.acl.label }} + spec: + containers: + - name: {{ .Values.acl.image.name }} + image: "{{ .Values.acl.image.name }}:{{ .Values.acl.image.tag }}" + env: + - name: KAFKA_BROKERS + value: {{ .Values.acl.kafka.brokers }} + - name: KAFKA_TOPIC + value: {{ .Values.acl.kafka.topic }} + - name: KAFKA_GROUP_ID + value: {{ .Values.acl.kafka.group }} + imagePullPolicy: {{ .Values.acl.pullPolicy }} + resources: + limits: + memory: {{ .Values.acl.resources.limits.memory }} + cpu: {{ .Values.acl.resources.limits.cpu }} + diff --git a/helm/templates/api/deployment.yml b/helm/templates/api/deployment.yml new file mode 100644 index 0000000..958e38e --- /dev/null +++ b/helm/templates/api/deployment.yml @@ -0,0 +1,32 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Values.api.label }}-deployment +spec: + selector: + matchLabels: + app: {{ .Values.api.label }} + template: + metadata: + labels: + app: {{ .Values.api.label }} + spec: + containers: + - name: {{ .Values.api.image.name }} + image: "{{ .Values.api.image.name }}:{{ .Values.api.image.tag }}" + env: + - name: DB_URI + value: {{ .Values.api.mongodb.uri }} + - name: KAFKA_BROKERS + value: {{ .Values.api.kafka.brokers }} + - name: KAFKA_TOPIC + value: {{ .Values.api.kafka.topic }} + - name: KAFKA_GROUP_ID + value: {{ .Values.api.kafka.group }} + imagePullPolicy: {{ .Values.api.pullPolicy }} + resources: + limits: + memory: {{ .Values.api.resources.limits.memory }} + cpu: {{ .Values.api.resources.limits.cpu }} + ports: + - containerPort: {{ .Values.api.port }} diff --git a/helm/templates/api/ingress.yml b/helm/templates/api/ingress.yml new file mode 100644 index 0000000..8ab992b --- /dev/null +++ b/helm/templates/api/ingress.yml @@ -0,0 +1,23 @@ +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: {{ .Values.api.label }}-ingress + labels: + name: {{ .Values.api.label }} +spec: + ingressClassName: {{ .Values.api.ingress.ingressClassName }} + rules: + - host: {{ .Values.api.ingress.host }} + http: + paths: + - pathType: Prefix + path: / + backend: + service: + name: {{ .Values.api.label }}-service + port: + number: {{ .Values.api.port }} + tls: + - hosts: + - {{ .Values.api.ingress.host }} + secretName: {{ .Values.api.label }}-secret diff --git a/helm/templates/api/service.yml b/helm/templates/api/service.yml new file mode 100644 index 0000000..c9add05 --- /dev/null +++ b/helm/templates/api/service.yml @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Values.api.label }}-service +spec: + selector: + app: {{ .Values.api.label }} + ports: + - port: {{ .Values.api.port }} + targetPort: {{ .Values.api.port }} \ No newline at end of file diff --git a/helm/templates/kafka/deployment.yml b/helm/templates/kafka/deployment.yml new file mode 100644 index 0000000..b8c8af1 --- /dev/null +++ b/helm/templates/kafka/deployment.yml @@ -0,0 +1,33 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Values.kafka.label }}-deployment +spec: + replicas: {{ .Values.kafka.replicas }} + selector: + matchLabels: + app: {{ .Values.kafka.label }} + template: + metadata: + labels: + app: {{ .Values.kafka.label }} + spec: + containers: + - env: + - name: KAFKA_BROKER_ID + value: "{{ .Values.kafka.variables.brokerId }}" + - name: KAFKA_ZOOKEEPER_CONNECT + value: {{ .Values.kafka.variables.zookeeperConnect }} + - name: KAFKA_LISTENERS + value: {{ .Values.kafka.variables.listeners }} + - name: KAFKA_ADVERTISED_LISTENERS + value: {{ .Values.kafka.variables.advertisedListeners }} + image: {{ .Values.kafka.image.name }} + imagePullPolicy: {{ .Values.kafka.pullPolicy }} + name: {{ .Values.kafka.label }} + resources: + limits: + memory: {{ .Values.kafka.resources.limits.memory }} + cpu: {{ .Values.kafka.resources.limits.cpu }} + ports: + - containerPort: {{ .Values.kafka.port }} \ No newline at end of file diff --git a/helm/templates/kafka/service.yml b/helm/templates/kafka/service.yml new file mode 100644 index 0000000..fbae7de --- /dev/null +++ b/helm/templates/kafka/service.yml @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Values.kafka.label }}-service +spec: + selector: + app: {{ .Values.kafka.label }} + ports: + - port: {{ .Values.kafka.port }} + targetPort: {{ .Values.kafka.port }} \ No newline at end of file diff --git a/helm/templates/mongodb/deployment.yml b/helm/templates/mongodb/deployment.yml new file mode 100644 index 0000000..40e3908 --- /dev/null +++ b/helm/templates/mongodb/deployment.yml @@ -0,0 +1,24 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Values.mongodb.label }}-deployment +spec: + selector: + matchLabels: + app: {{ .Values.mongodb.label }} + template: + metadata: + labels: + app: {{ .Values.mongodb.label }} + spec: + containers: + - name: {{ .Values.mongodb.image.name }} + image: "{{ .Values.mongodb.image.name }}:{{ .Values.mongodb.image.tag }}" + resources: + limits: + memory: {{ .Values.mongodb.resources.limits.memory }} + cpu: {{ .Values.mongodb.resources.limits.cpu }} + ports: + - containerPort: {{ .Values.mongodb.port }} + + diff --git a/helm/templates/mongodb/service.yml b/helm/templates/mongodb/service.yml new file mode 100644 index 0000000..b66fbaa --- /dev/null +++ b/helm/templates/mongodb/service.yml @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Values.mongodb.label }}-service +spec: + selector: + app: {{ .Values.mongodb.label }} + ports: + - port: {{ .Values.mongodb.port }} + targetPort: {{ .Values.mongodb.port }} \ No newline at end of file diff --git a/helm/templates/zookeeper/deployment.yml b/helm/templates/zookeeper/deployment.yml new file mode 100644 index 0000000..f0d843d --- /dev/null +++ b/helm/templates/zookeeper/deployment.yml @@ -0,0 +1,23 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: {{ .Values.zookeeper.label }}-deployment +spec: + selector: + matchLabels: + app: {{ .Values.zookeeper.label }} + template: + metadata: + labels: + app: {{ .Values.zookeeper.label }} + spec: + containers: + - image: {{ .Values.zookeeper.image.name }} + imagePullPolicy: {{ .Values.zookeeper.pullPolicy }} + name: {{ .Values.zookeeper.label }} + ports: + - containerPort: {{ .Values.zookeeper.port }} + resources: + limits: + memory: {{ .Values.zookeeper.resources.limits.memory }} + cpu: {{ .Values.zookeeper.resources.limits.cpu }} diff --git a/helm/templates/zookeeper/service.yml b/helm/templates/zookeeper/service.yml new file mode 100644 index 0000000..32996ee --- /dev/null +++ b/helm/templates/zookeeper/service.yml @@ -0,0 +1,10 @@ +apiVersion: v1 +kind: Service +metadata: + name: {{ .Values.zookeeper.label }}-service +spec: + selector: + app: {{ .Values.zookeeper.label }} + ports: + - port: {{ .Values.zookeeper.port }} + targetPort: {{ .Values.zookeeper.port }} \ No newline at end of file diff --git a/helm/values.yaml b/helm/values.yaml new file mode 100644 index 0000000..57dc2aa --- /dev/null +++ b/helm/values.yaml @@ -0,0 +1,80 @@ +mongodb: + label: mongodb + replicas: 1 + port: 27017 + image: + name: mongo + tag: latest + resources: + limits: + memory: "128Mi" + cpu: "500m" + +kafka: + label: kafka + replicas: 1 + port: 9092 + image: + name: wurstmeister/kafka + tag: latest + pullPolicy: IfNotPresent + resources: + limits: + memory: "512Mi" + cpu: "500m" + variables: + listeners: PLAINTEXT://:9092 + zookeeperConnect: zookeeper-service:2181 + advertisedListeners: PLAINTEXT://kafka-service:9092 + brokerId: "1" + +zookeeper: + label: zookeeper + replicas: 1 + port: 2181 + image: + name: wurstmeister/zookeeper + tag: latest + pullPolicy: IfNotPresent + resources: + limits: + memory: "128Mi" + cpu: "500m" + +acl: + label: acl + replicas: 5 + kafka: + brokers: kafka-service:9092 + topic: messages + group: acl + image: + name: acl + tag: latest + pullPolicy: Never + resources: + limits: + memory: "128Mi" + cpu: "500m" + +api: + label: api + replicas: 1 + port: 8080 + image: + name: api + tag: latest + pullPolicy: Never + resources: + limits: + memory: "128Mi" + cpu: "500m" + ingress: + host: foomyapp.com + ingressClassName: nginx + kafka: + brokers: kafka-service:9092 + topic: messages + group: acl + mongodb: + uri: mongodb://mongodb-service:27017/ \ No newline at end of file diff --git a/.dockerignore b/src/.dockerignore similarity index 100% rename from .dockerignore rename to src/.dockerignore diff --git a/Cargo.lock b/src/Cargo.lock similarity index 94% rename from Cargo.lock rename to src/Cargo.lock index 7da7f79..8a86bcd 100644 --- a/Cargo.lock +++ b/src/Cargo.lock @@ -76,7 +76,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e01ed3140b2f8d422c68afa1ed2e85d996ea619c988ac834d255db32138655cb" dependencies = [ "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -190,7 +190,7 @@ dependencies = [ "actix-router", "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -261,7 +261,7 @@ dependencies = [ ] [[package]] -name = "app" +name = "api" version = "0.0.1" dependencies = [ "actix-web", @@ -280,7 +280,7 @@ checksum = "a66537f1bb974b254c98ed142ff995236e81b9d0fe4db0575f46612cb15eb0f9" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -671,9 +671,9 @@ dependencies = [ [[package]] name = "env_logger" -version = "0.10.0" +version = "0.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "85cdab6a89accf66733ad5a1693a4dcced6aeff64602b634530dd73c1f3ee9f0" +checksum = "95b3f3e67048839cb0d0781f445682a35113da7121f7c949db0e2be96a4fbece" dependencies = [ "humantime", "is-terminal", @@ -690,9 +690,9 @@ checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" [[package]] name = "errno" -version = "0.3.5" +version = "0.3.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ac3e13f66a2f95e32a39eaa81f6b95d42878ca0e1db0c7543723dfe12557e860" +checksum = "f258a7194e7f7c2a7837a8913aeab7fd8c383457034fa20ce4dd3dcb813e8eb8" dependencies = [ "libc", "windows-sys", @@ -796,7 +796,7 @@ checksum = "53b153fd91e4b0147f4aced87be237c98248656bb01050b96bf3ee89220a8ddb" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -839,9 +839,9 @@ dependencies = [ [[package]] name = "getrandom" -version = "0.2.10" +version = "0.2.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be4136b2a15dd319360be1c07d9933517ccf0be8f16bf62a3bee4f0d618df427" +checksum = "fe9006bed769170c11f845cf00c7c1e9092aeb3f268e007c3e760ac68008070f" dependencies = [ "cfg-if", "libc", @@ -856,9 +856,9 @@ checksum = "6fb8d784f27acf97159b40fc4db5ecd8aa23b9ad5ef69cdd136d3bc80665f0c0" [[package]] name = "h2" -version = "0.3.21" +version = "0.3.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91fc23aa11be92976ef4729127f1a74adf36d8436f7816b185d18df956790833" +checksum = "4d6250322ef6e60f93f9a2162799302cd6f68f79f6e5d85c8c16f14d1d958178" dependencies = [ "bytes", "fnv", @@ -866,7 +866,7 @@ dependencies = [ "futures-sink", "futures-util", "http", - "indexmap 1.9.3", + "indexmap 2.1.0", "slab", "tokio", "tokio-util", @@ -917,7 +917,7 @@ name = "host" version = "0.0.1" dependencies = [ "actix-web", - "app", + "api", "async-trait", "domain", "domain_impl", @@ -944,9 +944,9 @@ dependencies = [ [[package]] name = "http" -version = "0.2.9" +version = "0.2.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd6effc99afb63425aff9b05836f029929e345a6148a14b7ecd5ab67af944482" +checksum = "8947b1a6fad4393052c7ba1f4cd97bed3e953a95c79c92ad9b051a04611d9fbb" dependencies = [ "bytes", "fnv", @@ -1147,9 +1147,9 @@ dependencies = [ [[package]] name = "js-sys" -version = "0.3.64" +version = "0.3.65" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c5f195fe497f702db0f318b07fdd68edb16955aed830df8363d837542f8f935a" +checksum = "54c0c35952f67de54bb584e9fd912b3023117cbafc0a77d8f3dee1fb5f572fe8" dependencies = [ "wasm-bindgen", ] @@ -1168,9 +1168,20 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" [[package]] name = "libc" -version = "0.2.149" +version = "0.2.150" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a08173bc88b7955d1b3145aa561539096c421ac8debde8cbc3612ec635fee29b" +checksum = "89d92a4743f9a61002fae18374ed11e7973f530cb3a3255fb354818118b2203c" + +[[package]] +name = "libredox" +version = "0.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85c833ca1e66078851dba29046874e38f08b2c883700aa29a03ddd3b23814ee8" +dependencies = [ + "bitflags 2.4.1", + "libc", + "redox_syscall", +] [[package]] name = "libz-sys" @@ -1192,9 +1203,9 @@ checksum = "0717cef1bc8b636c6e1c1bbdefc09e6322da8a9321966e8928ef80d20f7f770f" [[package]] name = "linux-raw-sys" -version = "0.4.10" +version = "0.4.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "da2479e8c062e40bf0066ffa0bc823de0a9368974af99c9f6df941d2c231e03f" +checksum = "969488b55f8ac402214f3f5fd243ebb7206cf82de60d3172994707a4bcc2b829" [[package]] name = "local-channel" @@ -1317,9 +1328,9 @@ dependencies = [ [[package]] name = "mongodb" -version = "2.7.0" +version = "2.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e22d517e7e678e1c9a2983ec704b43f3b22f38b1b7a247ea3ddb36d21578bf4e" +checksum = "e7c926772050c3a3f87c837626bf6135c8ca688d91d31dd39a3da547fc2bc9fe" dependencies = [ "async-trait", "base64 0.13.1", @@ -1437,9 +1448,9 @@ checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d" [[package]] name = "openssl" -version = "0.10.57" +version = "0.10.59" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bac25ee399abb46215765b1cb35bc0212377e58a061560d8b29b024fd0430e7c" +checksum = "7a257ad03cd8fb16ad4172fedf8094451e1af1c4b70097636ef2eac9a5f0cc33" dependencies = [ "bitflags 2.4.1", "cfg-if", @@ -1458,7 +1469,7 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -1469,9 +1480,9 @@ checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" [[package]] name = "openssl-sys" -version = "0.9.93" +version = "0.9.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "db4d56a4c0478783083cfafcc42493dd4a981d41669da64b4572a2a089b51b1d" +checksum = "40a4130519a360279579c2053038317e40eff64d13fd3f004f9e1b72b8a6aaf9" dependencies = [ "cc", "libc", @@ -1497,7 +1508,7 @@ checksum = "4c42a9226546d68acdd9c0a280d17ce19bfe27a46bf68784e4066115788d008e" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.4.1", + "redox_syscall", "smallvec", "windows-targets", ] @@ -1678,15 +1689,6 @@ dependencies = [ "pkg-config", ] -[[package]] -name = "redox_syscall" -version = "0.2.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" -dependencies = [ - "bitflags 1.3.2", -] - [[package]] name = "redox_syscall" version = "0.4.1" @@ -1698,12 +1700,12 @@ dependencies = [ [[package]] name = "redox_users" -version = "0.4.3" +version = "0.4.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b033d837a7cf162d7993aded9304e30a83213c648b6e389db233191f891e5c2b" +checksum = "a18479200779601e498ada4e8c1e1f50e3ee19deb0259c25825a98b5603b2cb4" dependencies = [ "getrandom", - "redox_syscall 0.2.16", + "libredox", "thiserror", ] @@ -1819,7 +1821,7 @@ dependencies = [ "quote", "rust-embed-utils", "shellexpand", - "syn 2.0.38", + "syn 2.0.39", "walkdir", ] @@ -1869,9 +1871,9 @@ dependencies = [ [[package]] name = "rustix" -version = "0.38.21" +version = "0.38.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2b426b0506e5d50a7d8dafcf2e81471400deb602392c7dd110815afb4eaf02a3" +checksum = "9ad981d6c340a49cdc40a1028d9c6084ec7e9fa33fcb839cab656a267071e234" dependencies = [ "bitflags 2.4.1", "errno", @@ -1882,9 +1884,9 @@ dependencies = [ [[package]] name = "rustls" -version = "0.21.8" +version = "0.21.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "446e14c5cda4f3f30fe71863c34ec70f5ac79d6087097ad0bb433e1be5edf04c" +checksum = "629648aced5775d558af50b2b4c7b02983a04b312126d45eeead26e7caa498b9" dependencies = [ "log", "ring", @@ -1894,9 +1896,9 @@ dependencies = [ [[package]] name = "rustls-pemfile" -version = "1.0.3" +version = "1.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2d3987094b1d07b653b7dfdc3f70ce9a1da9c51ac18c1b06b662e4f9a0e9f4b2" +checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c" dependencies = [ "base64 0.21.5", ] @@ -1997,9 +1999,9 @@ checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3" [[package]] name = "serde" -version = "1.0.190" +version = "1.0.192" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91d3c334ca1ee894a2c6f6ad698fe8c435b76d504b13d436f0685d648d6d96f7" +checksum = "bca2a08484b285dcb282d0f67b26cadc0df8b19f8c12502c13d966bf9482f001" dependencies = [ "serde_derive", ] @@ -2015,13 +2017,13 @@ dependencies = [ [[package]] name = "serde_derive" -version = "1.0.190" +version = "1.0.192" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "67c5609f394e5c2bd7fc51efda478004ea80ef42fee983d5c67a65e34f32c0e3" +checksum = "d6c7207fbec9faa48073f3e3074cbe553af6ea512d7c21ba46e434e70ea9fbc1" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -2132,9 +2134,9 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.11.1" +version = "1.11.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "942b4a808e05215192e39f4ab80813e599068285906cc91aa64f923db842bd5a" +checksum = "4dccd0940a2dcdf68d092b8cbab7dc0ad8fa938bf95787e1b916b0e3d0e8e970" [[package]] name = "socket2" @@ -2198,9 +2200,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.38" +version = "2.0.39" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e96b79aaa137db8f61e26363a0c9b47d8b4ec75da28b7d1d614c2303e232408b" +checksum = "23e78b90f2fcf45d3e842032ce32e3f2d1545ba6636271dcbf24fa306d87be7a" dependencies = [ "proc-macro2", "quote", @@ -2248,16 +2250,16 @@ checksum = "7ef1adac450ad7f4b3c28589471ade84f25f731a7a0fe30d71dfa9f60fd808e5" dependencies = [ "cfg-if", "fastrand", - "redox_syscall 0.4.1", + "redox_syscall", "rustix", "windows-sys", ] [[package]] name = "termcolor" -version = "1.3.0" +version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6093bad37da69aab9d123a8091e4be0aa4a03e4d601ec641c327398315f62b64" +checksum = "ff1bc3d3f05aff0403e8ac0d92ced918ec05b666a43f83297ccef5bea8a3d449" dependencies = [ "winapi-util", ] @@ -2268,7 +2270,7 @@ version = "0.0.1" dependencies = [ "actix-rt", "actix-web", - "app", + "api", "domain", "domain_impl", "host", @@ -2297,7 +2299,7 @@ checksum = "266b2e40bc00e5a6c09c3584011e08b06f123c00362c92b975ba9843aaaa14b8" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -2346,9 +2348,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.33.0" +version = "1.34.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4f38200e3ef7995e5ef13baec2f432a6da0aa9ac495b2c0e8f3b7eec2c92d653" +checksum = "d0c014766411e834f7af5b8f4cf46257aab4036ca95e9d2c144a10f59ad6f5b9" dependencies = [ "backtrace", "bytes", @@ -2365,13 +2367,13 @@ dependencies = [ [[package]] name = "tokio-macros" -version = "2.1.0" +version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "630bdcf245f78637c13ec01ffae6187cca34625e8c63150d424b59e55af2675e" +checksum = "5b8a1e28f2deaa14e508979454cb3a223b10b938b45af148bc0986de36f1923b" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -2569,9 +2571,9 @@ dependencies = [ [[package]] name = "utoipa" -version = "4.0.0" +version = "4.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6b208a50ff438dcdc887ea3f2db59530bd2f4bc3d2c70630e4d7ee7a281a1d1b" +checksum = "0ff05e3bac2c9428f57ade702667753ca3f5cf085e2011fe697de5bfd49aa72d" dependencies = [ "indexmap 2.1.0", "serde", @@ -2581,15 +2583,15 @@ dependencies = [ [[package]] name = "utoipa-gen" -version = "4.0.0" +version = "4.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0bd516d8879043e081537690bc96c8f17b5a4602c336aecb8f1de89d9d9c7e72" +checksum = "5f0b6f4667edd64be0e820d6631a60433a269710b6ee89ac39525b872b76d61d" dependencies = [ "proc-macro-error", "proc-macro2", "quote", "regex", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] @@ -2657,9 +2659,9 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" [[package]] name = "wasm-bindgen" -version = "0.2.87" +version = "0.2.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7706a72ab36d8cb1f80ffbf0e071533974a60d0a308d01a5d0375bf60499a342" +checksum = "7daec296f25a1bae309c0cd5c29c4b260e510e6d813c286b19eaadf409d40fce" dependencies = [ "cfg-if", "wasm-bindgen-macro", @@ -2667,24 +2669,24 @@ dependencies = [ [[package]] name = "wasm-bindgen-backend" -version = "0.2.87" +version = "0.2.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5ef2b6d3c510e9625e5fe6f509ab07d66a760f0885d858736483c32ed7809abd" +checksum = "e397f4664c0e4e428e8313a469aaa58310d302159845980fd23b0f22a847f217" dependencies = [ "bumpalo", "log", "once_cell", "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-futures" -version = "0.4.37" +version = "0.4.38" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c02dbc21516f9f1f04f187958890d7e6026df8d16540b7ad9492bc34a67cea03" +checksum = "9afec9963e3d0994cac82455b2b3502b81a7f40f9a0d32181f7528d9f4b43e02" dependencies = [ "cfg-if", "js-sys", @@ -2694,9 +2696,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro" -version = "0.2.87" +version = "0.2.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dee495e55982a3bd48105a7b947fd2a9b4a8ae3010041b9e0faab3f9cd028f1d" +checksum = "5961017b3b08ad5f3fe39f1e79877f8ee7c23c5e5fd5eb80de95abc41f1f16b2" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -2704,28 +2706,28 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.87" +version = "0.2.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b" +checksum = "c5353b8dab669f5e10f5bd76df26a9360c748f054f862ff5f3f8aae0c7fb3907" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", "wasm-bindgen-backend", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-shared" -version = "0.2.87" +version = "0.2.88" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ca6ad05a4870b2bf5fe995117d3728437bd27d7cd5f06f13c17443ef369775a1" +checksum = "0d046c5d029ba91a1ed14da14dca44b68bf2f124cfbaf741c54151fdb3e0750b" [[package]] name = "web-sys" -version = "0.3.64" +version = "0.3.65" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b85cbef8c220a6abc02aefd892dfc0fc23afb1c6a426316ec33253a3877249b" +checksum = "5db499c5f66323272151db0e666cd34f78617522fb0c1604d31a27c50c206a85" dependencies = [ "js-sys", "wasm-bindgen", @@ -2879,22 +2881,22 @@ dependencies = [ [[package]] name = "zerocopy" -version = "0.7.21" +version = "0.7.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "686b7e407015242119c33dab17b8f61ba6843534de936d94368856528eae4dcc" +checksum = "e97e415490559a91254a2979b4829267a57d2fcd741a98eee8b722fb57289aa0" dependencies = [ "zerocopy-derive", ] [[package]] name = "zerocopy-derive" -version = "0.7.21" +version = "0.7.26" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "020f3dfe25dfc38dfea49ce62d5d45ecdd7f0d8a724fa63eb36b6eba4ec76806" +checksum = "dd7e48ccf166952882ca8bd778a43502c64f33bf94c12ebe2a7f08e5a0f6689f" dependencies = [ "proc-macro2", "quote", - "syn 2.0.38", + "syn 2.0.39", ] [[package]] diff --git a/Cargo.toml b/src/Cargo.toml similarity index 94% rename from Cargo.toml rename to src/Cargo.toml index f62f385..d4a3a3a 100644 --- a/Cargo.toml +++ b/src/Cargo.toml @@ -1,7 +1,7 @@ [workspace] resolver = "2" members = [ - "app", + "api", "domain", "domain_impl", "host", diff --git a/acl/Cargo.toml b/src/acl/Cargo.toml similarity index 100% rename from acl/Cargo.toml rename to src/acl/Cargo.toml diff --git a/acl/Dockerfile b/src/acl/Dockerfile similarity index 83% rename from acl/Dockerfile rename to src/acl/Dockerfile index 4df613a..5329287 100644 --- a/acl/Dockerfile +++ b/src/acl/Dockerfile @@ -2,11 +2,14 @@ FROM rust:1.73.0-slim-buster as build WORKDIR /web-api COPY . . + +# install deps for building librdkafka RUN apt-get update && apt-get install -y build-essential \ openssl libssl-dev \ zlib1g \ cmake +# caching application build deps RUN --mount=type=cache,target=/web-api/target cargo build --release && cp target/release/acl /acl # acl image diff --git a/acl/src/lib.rs b/src/acl/src/lib.rs similarity index 100% rename from acl/src/lib.rs rename to src/acl/src/lib.rs diff --git a/acl/src/main.rs b/src/acl/src/main.rs similarity index 100% rename from acl/src/main.rs rename to src/acl/src/main.rs diff --git a/analyze.sh b/src/analyze.sh similarity index 100% rename from analyze.sh rename to src/analyze.sh diff --git a/app/Cargo.toml b/src/api/Cargo.toml similarity index 95% rename from app/Cargo.toml rename to src/api/Cargo.toml index 5952eec..17c8a42 100644 --- a/app/Cargo.toml +++ b/src/api/Cargo.toml @@ -1,5 +1,5 @@ [package] -name = "app" +name = "api" version = "0.0.1" edition = "2021" diff --git a/app/src/endpoints.rs b/src/api/src/endpoints.rs similarity index 100% rename from app/src/endpoints.rs rename to src/api/src/endpoints.rs diff --git a/app/src/lib.rs b/src/api/src/lib.rs similarity index 100% rename from app/src/lib.rs rename to src/api/src/lib.rs diff --git a/app/src/models.rs b/src/api/src/models.rs similarity index 100% rename from app/src/models.rs rename to src/api/src/models.rs diff --git a/build.sh b/src/build.sh similarity index 100% rename from build.sh rename to src/build.sh diff --git a/src/build_images.sh b/src/build_images.sh new file mode 100755 index 0000000..0b6dff2 --- /dev/null +++ b/src/build_images.sh @@ -0,0 +1,3 @@ +#!/bin/bash + +docker-compose --profile dev-build build \ No newline at end of file diff --git a/docker-compose.yml b/src/docker-compose.yml similarity index 93% rename from docker-compose.yml rename to src/docker-compose.yml index ad07079..04d880a 100644 --- a/docker-compose.yml +++ b/src/docker-compose.yml @@ -1,14 +1,8 @@ version: '3.8' services: - mongo: - hostname: mongo - container_name: mongo - image: mongo:7.0 - ports: - - "27017:27017" - acl: + profiles: ["dev-build"] build: context: . dockerfile: acl/Dockerfile @@ -25,13 +19,14 @@ services: - KAFKA_TOPIC=messages - KAFKA_GROUP_ID=dev - web-api: + api: + profiles: ["dev-build"] build: context: . dockerfile: host/Dockerfile - hostname: web-api - container_name: web-api - image: web-api:latest + hostname: api + container_name: api + image: api:latest depends_on: - kafka - mongo @@ -48,6 +43,13 @@ services: - KAFKA_TOPIC=messages - KAFKA_GROUP_ID=dev + mongo: + hostname: mongo + container_name: mongo + image: mongo:7.0 + ports: + - "27017:27017" + zookeeper: image: confluentinc/cp-zookeeper:7.0.1 hostname: zookeeper diff --git a/domain/Cargo.toml b/src/domain/Cargo.toml similarity index 100% rename from domain/Cargo.toml rename to src/domain/Cargo.toml diff --git a/domain/src/commands.rs b/src/domain/src/commands.rs similarity index 100% rename from domain/src/commands.rs rename to src/domain/src/commands.rs diff --git a/domain/src/events.rs b/src/domain/src/events.rs similarity index 100% rename from domain/src/events.rs rename to src/domain/src/events.rs diff --git a/domain/src/lib.rs b/src/domain/src/lib.rs similarity index 100% rename from domain/src/lib.rs rename to src/domain/src/lib.rs diff --git a/domain/src/queries.rs b/src/domain/src/queries.rs similarity index 100% rename from domain/src/queries.rs rename to src/domain/src/queries.rs diff --git a/domain_impl/Cargo.toml b/src/domain_impl/Cargo.toml similarity index 100% rename from domain_impl/Cargo.toml rename to src/domain_impl/Cargo.toml diff --git a/domain_impl/src/handlers.rs b/src/domain_impl/src/handlers.rs similarity index 100% rename from domain_impl/src/handlers.rs rename to src/domain_impl/src/handlers.rs diff --git a/domain_impl/src/lib.rs b/src/domain_impl/src/lib.rs similarity index 100% rename from domain_impl/src/lib.rs rename to src/domain_impl/src/lib.rs diff --git a/domain_impl/src/ports.rs b/src/domain_impl/src/ports.rs similarity index 100% rename from domain_impl/src/ports.rs rename to src/domain_impl/src/ports.rs diff --git a/format.sh b/src/format.sh similarity index 100% rename from format.sh rename to src/format.sh diff --git a/host/Cargo.toml b/src/host/Cargo.toml similarity index 94% rename from host/Cargo.toml rename to src/host/Cargo.toml index 3e776bb..b878b5f 100644 --- a/host/Cargo.toml +++ b/src/host/Cargo.toml @@ -13,7 +13,7 @@ async-trait = "0.1.73" utoipa = { version = "4", features = ["actix_extras"] } utoipa-swagger-ui = { version = "4", features = ["actix-web"] } -app = { path = "../app" } +api = { path = "../api" } domain = { path = "../domain" } domain_impl = { path = "../domain_impl" } infra = { path = "../infra" } diff --git a/host/Dockerfile b/src/host/Dockerfile similarity index 74% rename from host/Dockerfile rename to src/host/Dockerfile index c43f934..c6af9a3 100644 --- a/host/Dockerfile +++ b/src/host/Dockerfile @@ -2,14 +2,17 @@ FROM rust:1.73.0-slim-buster as build WORKDIR /web-api COPY . . + +# install deps for building librdkafka RUN apt-get update && apt-get install -y build-essential \ openssl libssl-dev \ zlib1g \ cmake +# caching application build deps RUN --mount=type=cache,target=/web-api/target cargo build --release && cp target/release/host /host -# app image -FROM debian:buster-slim as app +# api image +FROM debian:buster-slim as api COPY --from=build /host /host CMD ["/host"] \ No newline at end of file diff --git a/host/src/composition.rs b/src/host/src/composition.rs similarity index 94% rename from host/src/composition.rs rename to src/host/src/composition.rs index 4761ff2..5e7a247 100644 --- a/host/src/composition.rs +++ b/src/host/src/composition.rs @@ -9,8 +9,8 @@ use utoipa::OpenApi; use utoipa_swagger_ui::SwaggerUi; use crate::conf::AppConf; -use app::endpoints::{create_user, get_user, track_activity}; -use app::models::*; +use api::endpoints::{create_user, get_user, track_activity}; +use api::models::*; use domain::commands::{CreateUserCommand, ICommandHandler}; use domain::events::{ActivityEvent, IActivityTracker}; use domain::queries::{GetUserQuery, IQueryHandler, User}; @@ -22,9 +22,9 @@ use messaging::kafka::IKafkaFactory; #[derive(OpenApi)] #[openapi( paths( - app::endpoints::create_user, - app::endpoints::get_user, - app::endpoints::track_activity + api::endpoints::create_user, + api::endpoints::get_user, + api::endpoints::track_activity ), components( schemas(CreateUserRequest, CreatedUserIdResponse, ErrorResponse), diff --git a/host/src/conf.rs b/src/host/src/conf.rs similarity index 100% rename from host/src/conf.rs rename to src/host/src/conf.rs diff --git a/host/src/lib.rs b/src/host/src/lib.rs similarity index 100% rename from host/src/lib.rs rename to src/host/src/lib.rs diff --git a/host/src/main.rs b/src/host/src/main.rs similarity index 100% rename from host/src/main.rs rename to src/host/src/main.rs diff --git a/infra/Cargo.toml b/src/infra/Cargo.toml similarity index 100% rename from infra/Cargo.toml rename to src/infra/Cargo.toml diff --git a/infra/src/lib.rs b/src/infra/src/lib.rs similarity index 100% rename from infra/src/lib.rs rename to src/infra/src/lib.rs diff --git a/infra/src/repositories.rs b/src/infra/src/repositories.rs similarity index 100% rename from infra/src/repositories.rs rename to src/infra/src/repositories.rs diff --git a/infra/src/tracker.rs b/src/infra/src/tracker.rs similarity index 100% rename from infra/src/tracker.rs rename to src/infra/src/tracker.rs diff --git a/integration-tests/Cargo.toml b/src/integration-tests/Cargo.toml similarity index 95% rename from integration-tests/Cargo.toml rename to src/integration-tests/Cargo.toml index bb26c48..5152105 100644 --- a/integration-tests/Cargo.toml +++ b/src/integration-tests/Cargo.toml @@ -4,7 +4,7 @@ version = "0.0.1" edition = "2021" [dependencies] -app = { path = "../app" } +api = { path = "../api" } domain = { path = "../domain" } domain_impl = { path = "../domain_impl" } infra = { path = "../infra" } diff --git a/integration-tests/tests/gen.rs b/src/integration-tests/tests/gen.rs similarity index 89% rename from integration-tests/tests/gen.rs rename to src/integration-tests/tests/gen.rs index b679c7a..342f14c 100644 --- a/integration-tests/tests/gen.rs +++ b/src/integration-tests/tests/gen.rs @@ -1,4 +1,4 @@ -use app::models::TrackActivityRequest; +use api::models::TrackActivityRequest; use rand::Rng; use uuid::Uuid; diff --git a/integration-tests/tests/sut.rs b/src/integration-tests/tests/sut.rs similarity index 97% rename from integration-tests/tests/sut.rs rename to src/integration-tests/tests/sut.rs index 5db436e..86e8de9 100644 --- a/integration-tests/tests/sut.rs +++ b/src/integration-tests/tests/sut.rs @@ -1,5 +1,5 @@ use actix_web::rt; -use app::models::{CreateUserRequest, CreatedUserIdResponse, TrackActivityRequest, UserResponse}; +use api::models::{CreateUserRequest, CreatedUserIdResponse, TrackActivityRequest, UserResponse}; use domain::events::ActivityEvent; use host::composition::Composition; use host::conf::AppConf; @@ -47,7 +47,7 @@ impl Sut { match response.status() { StatusCode::OK => Ok(response.json().await.unwrap()), StatusCode::BAD_REQUEST => Err(response.text().await.unwrap()), - code => Err(format!("unexpected status code {}", code)) + code => Err(format!("unexpected status code {}", code)), } } @@ -68,7 +68,7 @@ impl Sut { match response.status() { StatusCode::CREATED => Ok(response.json().await.unwrap()), - code => Err(format!("unexpected status code {}", code)) + code => Err(format!("unexpected status code {}", code)), } } @@ -80,7 +80,7 @@ impl Sut { match response.status() { StatusCode::OK => Ok(()), - code => Err(format!("unexpected status code {}", code)) + code => Err(format!("unexpected status code {}", code)), } } diff --git a/integration-tests/tests/user_tests.rs b/src/integration-tests/tests/user_tests.rs similarity index 97% rename from integration-tests/tests/user_tests.rs rename to src/integration-tests/tests/user_tests.rs index d2307f9..349663c 100644 --- a/integration-tests/tests/user_tests.rs +++ b/src/integration-tests/tests/user_tests.rs @@ -1,7 +1,7 @@ mod gen; mod sut; -use app::models::TrackActivityRequest; +use api::models::TrackActivityRequest; use domain::events::ActivityEvent; use crate::gen::Gen; diff --git a/messaging/Cargo.toml b/src/messaging/Cargo.toml similarity index 100% rename from messaging/Cargo.toml rename to src/messaging/Cargo.toml diff --git a/messaging/src/fake_kafka.rs b/src/messaging/src/fake_kafka.rs similarity index 100% rename from messaging/src/fake_kafka.rs rename to src/messaging/src/fake_kafka.rs diff --git a/messaging/src/kafka.rs b/src/messaging/src/kafka.rs similarity index 100% rename from messaging/src/kafka.rs rename to src/messaging/src/kafka.rs diff --git a/messaging/src/lib.rs b/src/messaging/src/lib.rs similarity index 100% rename from messaging/src/lib.rs rename to src/messaging/src/lib.rs diff --git a/run.sh b/src/run.sh similarity index 100% rename from run.sh rename to src/run.sh diff --git a/run_tests.sh b/src/run_tests.sh similarity index 100% rename from run_tests.sh rename to src/run_tests.sh