Skip to content

Latest commit

 

History

History
223 lines (160 loc) · 5.33 KB

README.md

File metadata and controls

223 lines (160 loc) · 5.33 KB

LDAP-JWT Server

Lightweight node.js based web service that provides user authentication against LDAP server (Active Directory / Windows network) credentials and returns a JSON Web Token.

Features

  • Options to specify authorized groups in /authenticate, /verify
  • Option to bind either as the user attempting to authenticate or a service accout
  • Option to use SSL (httpS)

Usage

1. Configuration variables

Place configuration variables in .env file. Below is an example:

LDAP=enabled
LDAP_SEARCHFILTER=(cn={{username}})
LDAPAUTH_URL=ldaps://hostname
LDAPAUTH_SEARCHBASE=dc=example,dc=com
## If LDAPAUTH_BINDDN and LDAPAUTH_BINDCREDENTIALS are given,
## they will be used for binding to Active Directory.
LDAPAUTH_BINDDN=cn=binding,dc=example,dc=com
LDAPAUTH_BINDCREDENTIALS=secret
## Otherwise, authenticating user and corresponding credentials will be
## used to bind. LDAPAUTH_BINDDN_PREFIX and LDAPAUTH_BINDDN_SUFFIX must
## then be specified to flesh out the
## distinguished name (DN) of the user. For example:
# LDAPAUTH_BINDDN_PREFIX="cn="
# LDAPAUTH_BINDDN_SUFFIX=",ou=people,dc=example,dc=com"
CLIENT_ID=myclientid
CLIENT_SECRET=mysecret
SSL=true ## <-- turns on SSL (httpS)

2. SSL Certificates

If setting SSL=true, generate / obtain SSL certificates, and place the .crt and .key files in an 'ssl' directory under app: app/ssl/server.key and app/ssl/server.crt

3. Docker compose

See the docker-compose.yml file for configurations.

## build image:
docker compose build
## start container:
docker compose up

4. Verify server is running

Request with username and password:

$ curl -k -d '{"username":"<username>","password":"<password>"}' -H "Content-Type: application/json" -X POST "https://<hostname>/ldap-jwt/authenticate"

Request with username and password with authorized group (only allows access if username is in authorized group):

$ curl -k -d '{"username":"<username>","password":"<password>","authorized_groups":[<authorized group]}' -H "Content-Type: application/json" -X POST "https://<hostname>/ldap-jwt/authenticate"

Test Suite

Tests are written with jest and supertest. They can be run by setting the BUILD_TARGET environment variable to 'ci', building the image, and starting the container. Container exit status 0 indicates all tests passed.

docker compose build
docker compose up

Endpoints

/ldap-jwt/authenticate

No authorozed_groups

This example returns a token if <username> is any user in LDAP (and of course has the correct password).

Body of POST request:

{
    "username": <username>,
    "password": <password>
}

Response:

{
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "token": <JWT>
}

Payload of JWT:

{
  "aud": <CLIENT_ID from .env file>,
  "exp": <expiration date>,
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "user_name": <username>
}

w/ authorized_groups

This example returns a token if <username> is in LDAP and is a member of either <LDAP group1> or <LDAP group2> (and of course has the correct password).

Body of POST request:

{
    "username": <username>,
    "password": <password>,
    "authorized_groups": [ <LDAP group1>, <LDAP group2> ]
}

Note: requests that incorporate authorized_groups should only be made from the server side of an application. This practice 1) protects potentially sensitive LDAP group information and 2) prevents malicious users potentially gaining unauthorized access by substituting groups to which they belong.

Response:

{
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "token": <JWT>
}

Payload of JWT:

{
  "aud": <CLIENT_ID from .env file>,
  "exp": <expiration date>,
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "user_authorized_groups": [ <LDAP group1>, <LDAP groupA> ],
  "user_name": <username>
}  

In the JWT payload, the user_authorized_groups list is the intersection of the "authorized_groups" in the initial request and the user's groups in LDAP.

/ldap-jwt/verify

w/o authorized groups in token

In this example, the JWT payload does NOT contain an encoded authorized_groups key.

Body of POST request:

{
  "token": <JWT>
}

Response:

{
  "aud": <CLIENT_ID from .env file>,
  "exp": <expiration date>,
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "user_name": <username>
}

w/ authorized groups in token

In this example, the JWT payload contains an encoded authorized_groups key with values [ <LDAP group1>, <LDAP groupA> ].

Body of POST request:

{
  "token": <JWT>
}

Response:

{
  "aud": <CLIENT_ID from .env file>,
  "exp": <expiration date>,
  "full_name": <user's full name>,
  "mail": <user's email address>,
  "user_authorized_groups": [ <LDAP group1>, <LDAP groupA> ],
  "user_name": <username>
}

In the response, the user_authorized_groups list is the intersection of the "authorized_groups" in the original request made to generate the token and the user's groups in LDAP.

/ldap-jwt/health

A 'health check' endpoint that accepts GET requests and returns a 200 status and JSON message if the server is running.

Response:

{
  "message": "OK"
}