Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Auth Layer Server using KeyCloak #42

Merged
merged 5 commits into from
Mar 26, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,7 @@ charts/*.prov
charts/*.tar.gz
charts/hedera-the-graph/Chart.lock
charts/hedera-the-graph-node/Chart.lock
charts/auth-layer-server/Chart.lock

# DS_Store
.DS_Store
12 changes: 12 additions & 0 deletions charts/auth-layer-server/Chart.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
apiVersion: v2
name: auth-layer-server
appVersion: "19.1.0"
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Q: is this right? I thought we usually match the appVersion with the version.
Maybe you're making a suggestion to not do this

Copy link

@beeradb beeradb Mar 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yeah, since we are more or less mirroring the bitnami chart with some modified defaults, It seems like we should just also mirror the keycloak chart value for version and appVersion here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, this is right and on purpose, since appVersion normally refers to the version of the app, in this case is version 19.1.0 of the Bitnami Chart, however I also see your point and either way it would work and be fine.

description: A Helm chart for Authentication layer server for HederaTheGraph using Keycloak Bitnami chart as a base
type: application
version: 0.1.0
dependencies:
- alias: keycloak
name: keycloak
version: "19.1.0"
condition: keycloak.enabled
repository: "https://charts.bitnami.com/bitnami"
245 changes: 245 additions & 0 deletions charts/auth-layer-server/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,245 @@
# HederaTheGraph (HTG) Authentication Layer Server

The Authentication Layer Server is a KeyCloak server that provides authentication and authorization services for the HederaTheGraph (HTG). It is based on the [KeyCloak](https://www.keycloak.org/) open source identity and access management solution.
With the following configurations:
- Realm preconfigured with the HTG (htg-auth-layer) client
- Groups and Roles preconfigured for the HTG client
- Custom claim mappers for the HTG client
- Token expiration time set to 10 hours (plenty of time for a one day work session)
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: if there are links to docs explaining these settings we're overriding/modifying that would be useful here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes there is an official documentation for keycloak that we can link here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

✅ Done

- Disabled refresh tokens
- Other custom configurations
- User registration is disabled
- Forgot password is enabled
- Email as username is enabled
- Login with email is enabled

**For more information about KeyCloak Configuration, please visit the [KeyCloak Documentation](https://www.keycloak.org/docs/latest/server_admin/index.html)**

## Prerequisites
- Minikube or a Kubernetes cluster [(more here)](https://minikube.sigs.k8s.io/docs/start/)
- Helm 3 [(install instructions here)](https://helm.sh/docs/intro/install/)
- kubectl [(install instructions here)](https://kubernetes.io/docs/tasks/tools/)

## Installation

To install the Authentication Layer Server, run the following commands:

- Install the chart, using the custom values file provided in the `values.yaml` file
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: you also need to run helm dependency build if this is the first time using the chart (and presumably again if dependencies change?).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

✅ done

```bash
helm install htg-auth-server .
```

### Helm Chart Overrides

#### Admin Password

Define a custom password, use `auth.adminPassword` like this:
```bash
helm install htg-auth-server . --set auth.adminPassword=yourpassword
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This did not work for me - I installed via this command: helm install htg-auth-server . --set auth.adminPassword=secret.

I was able to successfully retrieve the admin password from the secret, so maybe it's just not setting the correct variable?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed, it was actually keycloak.auth.adminPassword=yourpassword

```

#### Client Secret

Define a custom client secret, use `auth.clientSecret` like this:
```bash
helm install htg-auth-server . --set auth.clientSecret=yourclientsecret
```

if setting the client secret from an umbrella chart and need to re-use the same client secret for other charts is recommended to use the `global.auth.clientSecret` property.
```yaml
global:
auth:
clientSecret: yourclientsecret
```

#### Use External DB (PostgreSQL)
By default as part of the helm chart, it will deploy a PostgreSQL database. If you want to use an external database, you can use the following configuration:

```yaml
keycloak:
postgresql:
enabled: false
externalDatabase:
host: myexternalhost
user: myuser
password: mypassword
database: mydatabase
port: 5432
```


*After installation please give some time (a few minutes) for the KeyCloak server to start and be ready to use.*

*For a whole list of configurable parameters see the oficial documentation of the [KeyCloak Helm Chart from Bitnami](https://artifacthub.io/packages/helm/bitnami/keycloak/12.1.3)*

### Configuration

#### Port Forwarding
You will need to port-forward the KeyCloak service to access the admin console. Run the following command to port-forward the service:
```bash
kubectl port-forward service/htg-auth-server-keycloak 8080:80
```
#### Login to the Admin Console
Access console on the following URK: `http://localhost:8080/admin/master/console/#/HederaTheGraph`
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: this might actually be better as a link. When following the readme I ended up just copying and pasting this URL.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

✅ done


Login with the following credentials:
- Username: `admin`
- Password: if not provided a random password is generated and can be found in the `htg-auth-server` secret with the following command:

```bash
kubectl get secret htg-auth-server-keycloak -o jsonpath="{.data.admin-password}" | base64 --decode; echo
```

#### Regenerate the Client Secret (optional)

The client secret is provided as a Value override using property `global.auth.clientSecret` or `clientSecret`, with global having precedence, and if neither is provided, a random secret is generated.

Is possible to see or regenerate the `htg-auth-layer` client secret from the KeyCloak admin console following these steps:

After successful login:
1. Select the `HederaTheGraph` realm from the dropdown menu.
2. Then navigate to `Clients` and select the `htg-auth-layer` client.
3. Click on the `Credentials` tab and
4. Click on the `Regenerate` butto of `Client Secret` section to generate a new client secret.
5. Copy the new client secret since that will be needed to requests tokens from the Authentication Layer Server and to call the instrospection endpoint.

#### Configure Email service

The Auth server will need to send emails for the forgot password feature, and verify email. The service is configured following the next steps:
1. Login to the KeyCloak admin console.
2. Select the `HederaTheGraph` realm from the dropdown menu.
3. Then navigate to `Realm Settings` and select the `Email` tab.
4. Fill in the email settings and click on the `Save` button.

Uses `smtp` host, port and optionally credentials (username and password) to send emails.
Is also required to confiture `From` and recommended to configure `From Display Name` fields.

*Once the email service is configured, is possible to test the email service by sending a test email from the `Email` tab to verify that the email service is working.*


### Usage

#### Create a new user

Once you have your KeyCloak server running, you can create a new user by following these steps:

1. Login to the KeyCloak admin console.
2. Select the `HederaTheGraph` realm from the dropdown menu (if not selected already).
3. Then navigate to `Users` and click on the `Add user` button.
4. Fill in the user details and click on the `Save` button.
5. After saving the user, navigate to the `Credentials` tab and set a password for the user (prefferably a temporary one, that forces the user to update it at next login).
6. Assign Roles or Grpups to the user.
7. Set `subgraph_access` attribute to the user, this will give the user access to these subgraphs. This is done by click the `Attributes` tab and adding a new attribute with the name `subgraph_access` and the value being the subgraph id. The value can be a CSV string with many subgraph names, ie: `subgraph1,subgraph2,subgraph3`.

#### Groups and Roles

**a. Subgraph Admin:**, has access to all the subgraphs admin endpoints. (`subgraph_create`, `subgraph_remove`, `subgraph_deploy`,`subgraph_pause` and `subgraph_resume`)
**b. Subgraph Deployer:**, has access to: `subgraph_pause`, `subgraph_resume` and `subgraph_deploy` endpoints.
**c. Subgraph Operator:**, has access to: `subgraph_pause` and 'subgraph_resume' endpoints.

#### Get a Token from the Authentication Layer Server

If using port-forwarding, you can get a token from the Authentication Layer Server by running the following command, you will need the following information:
- `username` or `email` of the user
- `user-password` of the user
- `client-secret` of the `htg-auth-layer` client


```bash
curl --location 'http://localhost:8080/realms/HederaTheGraph/protocol/openid-connect/token' \
--header 'Content-Type: application/x-www-form-urlencoded' \
--data-urlencode 'grant_type=password' \
--data-urlencode 'username=<username or email>' \
--data-urlencode 'password=<user-password>' \
--data-urlencode 'client_id=htg-auth-layer' \
--data-urlencode 'client_secret=<client-secret>'
```

*However is recommended to expose the service using a FQDN since the token that is created using the host `localhost` is only valid for the `localhost` host when calling the instrospection endpoint. So port-forwarding is only recommended for testing purposes.*


#### Verify a Token
To verify a token, you can call the introspection endpoint of the Authentication Layer Server. The introspection endpoint is available at the following URL: `http://localhost:8080/auth/realms/HederaTheGraph/protocol/openid-connect/token/introspect`
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: this also would probably be better as a link

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is not mean to be called by a browser, so the link does not makes too much sense.


```bash
curl --location 'http://localhost:8080/realms/HederaTheGraph/protocol/openid-connect/token/introspect' \
--header 'Content-Type: application/x-www-form-urlencoded' \
--data-urlencode 'token=<access-token>' \
--data-urlencode 'client_id=htg-auth-layer' \
--data-urlencode 'client_secret=<client-secret>'
```

if the token is valid, the response will be a JSON object with the token information, otherwise, the response will be a JSON object with the `active` property set to `false`.

**Successful example:**

Verify that it contains the assigned roles, and the desired subgraph_access.

```json
{
"exp": 1709785194,
"iat": 1709749194,
"jti": "b61662b4-e3b5-4009-bc08-a7821fcfa2ee",
"iss": "http://localhost:8080/realms/HederaTheGraph",
"aud": "account",
"sub": "6e942cb3-4e82-447b-ac0c-641c764ef8f3",
"typ": "Bearer",
"azp": "htg-auth-layer",
"session_state": "99891cd0-dc7f-44fd-9187-34f30f57e5d0",
"acr": "1",
"allowed-origins": [
"/*"
],
"realm_access": {
"roles": [
"default-roles-hederathegraph",
"subgraph_remove",
"subgraph_create",
"subgraph_deploy",
"offline_access",
"uma_authorization",
"subgraph_resume",
"subgraph_pause"
]
},
"resource_access": {
"account": {
"roles": [
"manage-account",
"manage-account-links",
"view-profile"
]
}
},
"scope": "profile subgraph_access email",
"sid": "99891cd0-dc7f-44fd-9187-34f30f57e5d0",
"email_verified": true,
"name": "User name",
"subgraph_access": "subgraph1,subgraph2",
"preferred_username": "[email protected]",
"given_name": "User First Name",
"family_name": "User Last Name",
"email": "[email protected]",
"client_id": "htg-auth-layer",
"username": "[email protected]",
"token_type": "Bearer",
"active": true
}
```

**Unsuccessful example: (invalid token or expired)**

```json
{
"active": false
}
```

**Unsuccessful example: (invalid client_id or client_secret)**

```json
{
"error": "invalid_request",
"error_description": "Authentication failed."
}
```
12 changes: 12 additions & 0 deletions charts/auth-layer-server/templates/_helpers.tpl
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
{{/*
Define a function to choose a value from .Values.global.auth.clientSecret, .Values.clientSecret, or generate a random string.
*/}}
{{- define "auth.clientSecret" -}}
{{- if .Values.global.auth.clientSecret -}}
{{- .Values.global.auth.clientSecret -}}
{{- else if .Values.clientSecret -}}
{{- .Values.clientSecret -}}
{{- else -}}
{{- randAlphaNum 32 -}}
{{- end -}}
{{- end -}}
Loading
Loading