Broken Crystals is a benchmark application that uses modern technologies and implements a set of common security vulnerabilities.
The application contains:
- React based web client
- FE - http://localhost:8090
- BE - http://localhost:3000
- NodeJS server - the full API documentation is available via swagger
- Swagger UI - http://localhost:8090/swagger
- Swagger JSON file - http://localhost:8090/swagger-json
- nginx web server that serves the client and acts as a reverse proxy for the server's API requests
# build server
npm ci && npm run build
# build client
npm ci --prefix public && npm run build --prefix public
#build and start dockers with Postgres DB, nginx and server
docker-compose --file=docker-compose.local.yml up -d
#rebuild dockers
docker-compose --file=docker-compose.local.yml up -d --build
Running tests by SecTester
In the path ./test
you can find tests to run with Jest.
First, you have to get a Bright API key, navigate to your .env
file, and paste your Bright API key as the value of the BRIGHT_TOKEN
variable:
BRIGHT_TOKEN = <your_API_key_here>
Then, you can modify a URL to your instance of the application by setting the SEC_TESTER_TARGET
environment variable in your .env
file:
SEC_TESTER_TARGET = http://localhost:8090
Finally, you can start tests with SecTester against these endpoints as follows:
npm run test:e2e
Full configuration & usage examples can be found in our demo project;
-
Broken JWT Authentication - The application includes multiple endpoints that generate and validate several types of JWT tokens. The main login API, used by the UI, is utilizing one of the endpoints while others are available via direct call and described in Swagger.
- No Algorithm bypass - Bypasses the JWT authentication by using the “None” algorithm (implemented in main login and API authorization code).
- RSA to HMAC - Changes the algorithm to use a “HMAC” variation and signs with the public key of the application to bypass the authentication (implemented in main login and API authorization code).
- Invalid Signature - Changes the signature of the JWT to something different and bypasses the authentication (implemented in main login and API authorization code).
- KID Manipulation - Changes the value of the KID field in the Header of JWT to use either: (1) a static file that the application uses or (2) OS Command that echoes the key that will be signed or (3) SQL code that will return a key that will be used to sign the JWT (implemented in designated endpoint as described in Swagger).
- Brute Forcing Weak Secret Key - Checks if common secret keys are used (implemented in designated endpoint as described in Swagger). The secret token is configurable via .env file and, by default, is 123.
- X5U Rogue Key - Uses the uploaded certificate to sign the JWT and sets the X5U Header in JWT to point to the uploaded certificate (implemented in designated endpoint as described in Swagger).
- X5C Rogue Key - The application doesn’t properly check which X5C key is used for signing. When we set the X5C headers to our values and sign with our private key, authentication is bypassed (implemented in designated endpoint as described in Swagger).
- JKU Rogue Key - Uses our publicly available JSON to check if JWT is properly signed after we set the Header in JWT to point to our JSON and sign the JWT with our private key (implemented in designated endpoint as described in Swagger).
- JWK Rogue Key - We make a new JSON with empty values, hash it, and set it directly in the Header and we then use our private key to sign the JWT (implemented in designated endpoint as described in Swagger).
-
Brute Force Login - Checks if the application user is using a weak password. The default setup contains user = admin with password = admin
-
Common Files - Tries to find common files that shouldn’t be publicly exposed (such as “phpinfo”, “.htaccess”, “ssh-key.priv”, etc…). The application contains .htacess and Nginx.conf files under the client's root directory and additional files can be added by placing them under the public/public directory and running a build of the client.
-
Cookie Security - Checks if the cookie has the “secure” and HTTP only flags. The application returns two cookies (session and bc-calls-counter cookie), both without secure and HttpOnly flags.
-
Cross-Site Request Forgery (CSRF)
- Checks if a form holds anti-CSRF tokens, misconfigured “CORS” and misconfigured “Origin” header - the application returns "Access-Control-Allow-Origin: *" header for all requests. The behavior can be configured in the /main.ts file.
- The same form with both authenticated and unauthenticated user - the Email subscription UI forms can be used for testing this vulnerability.
- Different form for an authenticated and unauthenticated user - the Add testimonial form can be used for testing. The forms are only available to authenticated users.
-
Cross-Site Scripting (XSS) -
- Reflective XSS can be demonstrated by using the mailing list subscription form on the landing page.
- Persistent XSS can be demonstrated using add testimonial form on the landing page (for authenticated users only).
-
Default Login Location - The login endpoint is available under /api/auth/login.
-
Directory Listing - The Nginx config file under the nginx-conf directory is configured to allow directory listing.
-
DOM Cross-Site Scripting - Open the landing page with the dummy query param that contains DOM content (including script), add the provided DOM into the page, and execute it.
-
File Upload - The application allows uploading an avatar photo of the authenticated user. The server doesn't perform any sort of validation on the uploaded file.
-
Full Path Disclosure - All errors returned by the server include the full path of the file where the error has occurred. The errors can be triggered by passing wrong values as parameters or by modifying the bc-calls-counter cookie to a non-numeric value.
-
Headers Security Check - The application is configured with misconfigured security headers. The list of headers is available in the headers.configurator.interceptor.ts file. A user can pass the no-sec-headers query param to any API to prevent the server from sending the headers.
-
HTML Injection - Both forms testimonial and mailing list subscription forms allow HTML injection.
-
HTTP Method fuzzer - The server supports uploading, deletion, and getting the content of a file via /put.raw addition to the URL. The actual implementation using a regular upload endpoint of the server and the /put.raw endpoint is mapped in Nginx.
-
LDAP Injection - The login request returns an LDAP query for the user's profile, which can be used as a query parameter in /api/users/ldap query query parameter. The returned query can be modified to search for other users. If the structure of the LDAP query is changed, a detailed LDAP error will be returned (with LDAP server information and hierarchy).
-
Local File Inclusion (LFI) - The /api/files endpoint returns any file on the server from the path that is provided in the path param. The UI uses this endpoint to load crystal images on the landing page.
-
Mass Assignment - You can add to user admin privilegies upon creating user or updating userdata. When you creating a new user /api/users/basic you can use additional hidden field in body request { ... "isAdmin" : true }. If you are trying to edit userdata with PUT request /api/users/one/{email}/info you can add this additional field mentioned above. For checking admin permissions there is one more endpoint: /api/users/one/{email}/adminpermission.
-
Open Database - The index.html file includes a link to manifest URL, which returns the server's configuration, including a DB connection string.
-
OS Command Injection - The /api/spawn endpoint spawns a new process using the command in the command query parameter. The endpoint is not referenced from UI.
-
Remote File Inclusion (RFI) - The /api/files endpoint returns any file on the server from the path that is provided in the path param. The UI uses this endpoint to load crystal images on the landing page.
-
Secret Tokens - The index.html file includes a link to manifest URL, which returns the server's configuration, including a Google API key.
-
Server-Side Template Injection (SSTI) - The endpoint /api/render receives a plain text body and renders it using the doT (http://github.com/olado/dot) templating engine.
-
Server-Side Request Forgery (SSRF) - The endpoint /api/file receives the path and type query parameters and returns the content of the file in path with Content-Type value from the type parameter. The endpoint supports relative and absolute file names, HTTP/S requests, as well as metadata URLs of Azure, Google Cloud, AWS, and DigitalOcean.
-
SQL injection (SQLI) - The /api/testimonials/count endpoint receives and executes SQL query in the query query parameter.
-
Unvalidated Redirect - The endpoint /api/goto redirects the client to the URL provided in the url query parameter. The UI references the endpoint in the header (while clicking on the site's logo) and as an href source for the Terms and Services link in the footer.
-
Version Control System - The client_s build process copies SVN, GIT, and Mercurial source control directories to the client application root and they are accessible under Nginx root.
-
XML External Entity (XXE) - The endpoint, POST /api/metadata, receives URL-encoded XML data in the xml query parameter, processes it with enabled external entities (using libxmnl library) and returns the serialized DOM. Additionally, for a request that tries to load file:///etc/passwd as an entity, the endpoint returns a mocked up content of the file.
-
JavaScript Vulnerabilities Scanning - Index.html includes an older version of the jQuery library with known vulnerabilities.