Skip to content

Commit

Permalink
[#16] Implementing security for api-server
Browse files Browse the repository at this point in the history
The API Server provides a security model that provides authentication
and authorization of incoming clients.

The security can be enabled/disabled (i.e. via `API_SERVER_SECURITY_ENABLED` env var)

Currently the api server support `jwt` token authentication.

The login api is defined in openapi.yml

/server/login

A client logs in to an api server by sending a POST request to the login path.
The request body contains login information (i.e. username and password for jwt authentication type)

Please refer to [api.md](api.md) for details of the login api.

Currently the security manager uses local file to store user's info.
The default users file name is `.users.json`
The users file name can be configured using `USERS_FILE_URL` env var.
See `.test.users.json` for sample values.

Currently the api server doesn't perform authorization on logged in users.

The server keeps a list of jolokia endpoints for clients to access.
The endpoints are loaded from a local file named
`.endpoints.json`. Each top level entry represents a jolokia endpoint.
An entry has a unique name and details to access the jolokia api.

See `.test.endpoints.json` for sample values.

When an authenticated client sends a request to the api-server, it
should present its token in the request header

    'Authorization: Bearer `token`'

It also need to give the `targetEndpoint` in the query part of the request
if the request is to access an jolokia endpoint.

For example `/execBrokerOperation?targetEndpoint=broker1`.

Direct Proxy means a client can pass a broker's endpoint info to the
api-server in order to access it via the api-server.

For example the [self-provisioning plugin]
(https://github.com/artemiscloud/activemq-artemis-self-provisioning-plugin)
uses this api to access the jolokia of a broker's jolokia endpoint.
  • Loading branch information
howardgao committed Dec 2, 2024
1 parent f65dc48 commit e67fbb9
Show file tree
Hide file tree
Showing 35 changed files with 2,499 additions and 204 deletions.
12 changes: 12 additions & 0 deletions .dockerignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
.git/
.gitignore
node_modules/
dist/
*.md
logs/
tmp
.eslintrc.yml
.prettierrc.yml
.test*
.jest.config.js
.users.json
16 changes: 10 additions & 6 deletions .env
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,16 @@ PLUGIN_NAME='ActiveMQ Artemis Jolokia api-server'
SERVER_CERT=/var/serving-cert/tls.crt
SERVER_KEY=/var/serving-cert/tls.key

# replace the token in production deployment
SECRET_ACCESS_TOKEN=1e13d44f998dee277deae621a9012cf300b94c91

# to trust jolokia certs
NODE_TLS_REJECT_UNAUTHORIZED='0'

# logging
LOG_LEVEL='info'
ENABLE_REQUEST_LOG='false'

# security

# replace the token in production deployment
SECRET_ACCESS_TOKEN=1e13d44f998dee277deae621a9012cf300b94c91

API_SERVER_SECURITY_ENABLED=true
API_SERVER_SECURITY_AUTH_TYPE=jwt
USERS_FILE_URL=.users.json
ENDPOINTS_FILE_URL=.endpoints.json
5 changes: 5 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -121,3 +121,8 @@ dist
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.*

# vs code config
.vscode


57 changes: 57 additions & 0 deletions .test.endpoints.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
{
"endpoints": [
{
"name": "broker1",
"url": "http://127.0.0.1:8161",
"auth": [
{
"scheme": "basic",
"data": {
"username": "guest",
"password": "b5e722c9d28806fe552f3c8f7ec22b78:93676670ebddead3b9c18c097f9aa1b2"
}
}
]
},
{
"name": "broker2",
"url": "http://127.0.0.2:8161",
"auth": [
{
"scheme": "basic",
"data": {
"username": "guest",
"password": "c9197017fb498a03650f49c4301036b5:17abb83fa3ec12b966889a6f76800e8a"
}
}
]
},
{
"name": "broker3",
"url": "http://127.0.0.3:8161",
"auth": [
{
"scheme": "basic",
"data": {
"username": "admin",
"password": "a61fa665c0cf11109f53645d5aca382e:e6cffc14526c63f51a359afdc8b7065d"
}
}
]
},
{
"name": "broker4",
"url": "https://artemis-broker-jolokia-0-svc-ing-default.artemiscloud.io:443",
"jolokiaPrefix": "/jolokia/",
"auth": [
{
"scheme": "cert",
"data": {
"certpath": "test-api-server.crt",
"keypath": "test-api-server.key"
}
}
]
}
]
}
22 changes: 22 additions & 0 deletions .test.env
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@

PLUGIN_VERSION=1.0.0
PLUGIN_NAME='ActiveMQ Artemis Jolokia api-server'

# dev cert
SERVER_CERT=/var/serving-cert/tls.crt
SERVER_KEY=/var/serving-cert/tls.key

# replace the token in production deployment
SECRET_ACCESS_TOKEN=1e13d44f998dee277deae621a9012cf300b94c91

# to trust jolokia certs
NODE_TLS_REJECT_UNAUTHORIZED='0'

# logging
LOG_LEVEL='debug'
ENABLE_REQUEST_LOG=false

# security
API_SERVER_SECURITY_ENABLED=true
USERS_FILE_URL=.test.users.json
ENDPOINTS_FILE_URL=.test.endpoints.json
26 changes: 26 additions & 0 deletions .test.users.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
{
"users": [
{
"id": "user1",
"email": "[email protected]",
"hash": "$2a$12$nv9iV5/UNuV4Mdj1Jf8zfuUraqboSRtSQqCmtOc4F7rdwmOb9IzNu"
},
{
"id": "user2",
"hash": "$2a$12$VHZ9aJ5A87YeFop4xVW.aOMm95ClU.EviyT9o0i8HYLdG6w6ctMfW"
},
{
"id": "root",
"email": "[email protected]",
"hash": "$2a$12$VHZ9aJ5A87YeFop4xVW.aOMm95ClU.EviyT9o0i8HYLdG6w6ctMfW"
},
{
"id": "usernoroles",
"hash": "$2a$12$J.CGc062y9YhdvYEqqiqoetDronJwUFKR0f2fGARwgHuasAt/QKa2"
},
{
"id": "admin",
"hash": "$2a$15$deBI0sO4Cvn2gr/QA5pju.94Klh377Np.mcYteBgYlQuyeIwyK4UK"
}
]
}
8 changes: 8 additions & 0 deletions .users.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"users": [
{
"id": "admin",
"hash": "$2a$15$deBI0sO4Cvn2gr/QA5pju.94Klh377Np.mcYteBgYlQuyeIwyK4UK"
}
]
}
60 changes: 58 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ yarn run build-api-doc
### deploy the service

```sh
./deploy.sh [-i <image> -n]
./deploy.sh [-i <image> -ns]
```

The optional `-i <image>` (or `--image <image>`) argument allows you to pass in
Expand All @@ -48,6 +48,15 @@ deployed. for example:
```sh
./deploy.sh -i quay.io/<repo-username>/activemq-artemis-jolokia-api-server:1.0.1
```
The optional -ns (or --nosec) argument can be used to disable security.

---
**Note:**

you should enable security in your application. Disable security can only
be used for test purposes.

---

The `deploy.sh` script uses `oc kustomize` (built-in
[kustomize](https://github.com/kubernetes-sigs/kustomize)) command to configure
Expand All @@ -67,6 +76,53 @@ jwt tokens. It has a default value in .env for dev purposes.

In production you should override it with your own secret.

The jwt-key-gen.sh is a tool to generate a random key and used in Dockerfile.
The jwt-key-gen.sh is a tool to generate a random key and used in Dockerfile.
It makes sure when you build the api server image a new random key is used.

## Security Model of the API Server

The API Server provides a security model that provides authentication and authorization of incoming clients.
The security can be enabled/disabled (i.e. via `API_SERVER_SECURITY_ENABLED` env var)

### Authentication

Currently the api server support `jwt` token authentication.

#### The login api

The login api is defined in openapi.yml

```yaml
/server/login
```

A client logs in to an api server by sending a POST request to the login path. The request body contains login information (i.e. username and password for jwt authentication type)

Please refer to [api.md](api.md) for details of the log api.

Currently the security manager uses local file to store user's info. The default users file name is `.users.json`
The users file name can be configured using `USERS_FILE_URL` env var. See `.test.users.json` for sample values.

### Authorization

Currently the api server doesn't perform authorization on logged in users.

### Endpoints Management

The server keeps a list of jolokia endpoints for clients to access. The endpoints are loaded from a local file named
`.endpoints.json`. Each top level entry represents a jolokia endpoint. An entry has a unique name and details to access the jolokia api. See `.test.endpoints.json` for sample values.

### Accessing a jolokia endpoint

When an authenticated client sends a request to the api-server, it should present its token in the request header

'Authorization: Bearer `token`'

It also need to give the `targetEndpoint` in the query part of the request if the request is to access an jolokia endpoint.

For example `/execBrokerOperation?targetEndpoint=broker1`.

### Direct Proxy

Direct Proxy means a client can pass a broker's endpoint info to the api-server in order to access it via the api-server.
For example the [self-provisioning plugin](https://github.com/artemiscloud/activemq-artemis-self-provisioning-plugin) uses this api to access the jolokia of a broker's jolokia endpoint.
Loading

0 comments on commit e67fbb9

Please sign in to comment.