-
Notifications
You must be signed in to change notification settings - Fork 54
Web Application Firewall (WAF) Troubleshooting and Maintenance
SimpleReport is, by its nature, a dynamic application that grows and changes with each and every development sprint. As the application changes, so to does its userbase, in response to the rising and falling tides of infectious disease spread.
As the stewards of PII and confidential healthcare information, we must undertake due diligence to protect the data with which we are entrusted. In doing so, however, we must not make the application so secure that its functionality is unduly restricted for the average user. In its current configuration, the WAF has been tuned to permit the vast majority of user functionality; however, unforeseen cases may arise in which a user’s routine transactions are blocked.
This guide is intended to diagnose and provide mitigation instructions for these instances.
If a transactional request is blocked by the WAF, it will manifest in the following manner:
The user will typically be presented with a generic error message:
Unexpected token < in JSON at position 0
This message will also usually be accompanied by a more user-friendly failure message.
Analysis of the browser console and associated application logs will show an HTTP response code 403 ("Forbidden").
For confirmation, please collect the following information:
- Which screen the user was on at the time of the error
- The approximate timestamp of the error
- Any pertinent information about the user that would allow DevOps personnel to identify the problematic request. If this is PII, please ensure it is properly safeguarded when communicating.
To confirm involvement of the WAF, within Azure, navigate to the simple-report-app-gateway
Application Gateway, housed in the prime-simple-report-prod
resource group. In the left-side menu, under Monitoring, select Logs.
Run the following query to pull results from the WAF:
AzureDiagnostics
| where ResourceProvider == "MICROSOFT.NETWORK" and Category == "ApplicationGatewayFirewallLog"
Unfortunately, we are not currently able to lift exceptions on the basis of field content or the content of query string arguments. Likewise, we are not able to allowlist specific users, diseases, etc.
When developing new features and functionality, the goal is to develop in a manner that preserves as much security as possible. If changes to the WAF are necessary, they should be small enough to permit the largest amount of blocking possible, while still allowing for the new feature to function as intended. This same principle should be adhered to in the event that a problem is discovered in production.
At present, there are four different items that can be excepted: Cookies, Headers, Arguments, and entire Rules. Only three are actively used in the WAF's current implementation; instructions for each follow.
Knowledge of Terraform is required to make the following changes. For assistance, please contact the DevOps team.
The WAF operates using the OWASP 3.1 rule set as a baseline. Exceptions consist of a rule_group_override
, which is further broken down by rule_group_name
, and a list of disabled_rules
:
rule_group_override {
rule_group_name = "REQUEST-920-PROTOCOL-ENFORCEMENT"
disabled_rules = [
"920300",
"920320"
]
}
All rule_group_override
blocks are daughters of the managed_rule_set
block.
To add a new rule, first gather the rule group and rule ID from the Application Gateway logs. Ensure that the associated rule group is present in a rule_group_override
block, and simply add a new entry in the disabled_rules
array.
If a new rule_group_override
block must be added, ensure you follow the syntax as it exists in the file.
Occasionally, changes to application structure or third-party integrations will necessitate the addition of a cookie exception. The Application Gateway logs will indicate when this is necessary by providing information about which cookie tripped the firewall.
To create a cookie exception, you will need to create a new exclusion
block, like so:
exclusion {
match_variable = "RequestCookieNames"
selector = "ai_session" //Part of Azure Application Insights
selector_match_operator = "StartsWith"
}
The match_variable
will always be RequestCookieNames
for cookie exceptions. Adjust the values for selector
and selector_match_operator
to properly match the data given to you by the Application Gateway logs.
All exclusion
blocks are daughters of the managed_rules
superblock.
Occasionally, changes to application structure, or an unfortunate combination of user data with an overly-sensitive security rule, will require entire arguments within a GraphQL query or other query string to be excepted. (Fun fact: the example below was added because any user or facility address with the word "Union" in the street name made the WAF think an active SQL injection attempt was ongoing.) The Application Gateway logs will indicate when this is necessary by providing information about which argument tripped the firewall. ()
A good indicator that an argument exception will need to be applied is if the triggering request was made against the /api/graphql
endpoint. Additionally, the keyword "ARGS" will make a prominent appearance in the log entry.
To create an argument exception, you will need to create a new exclusion
block, like so:
exclusion {
match_variable = "RequestArgNames"
selector = "variables.street"
selector_match_operator = "Equals"
}
The match_variable
will always be RequestArgNames
for argument exceptions. Adjust the values for selector
and selector_match_operator
to properly match the data given to you by the Application Gateway logs.
All exclusion
blocks are daughters of the managed_rules
superblock.
Help and support will be needed from all members of the SimpleReport team to keep the WAF functioning at its best, without disrupting the flow of the application.
If you are a member of any of the following groups, we ask for your support in executing the following tasks:
- Test any changes to query strings, new features, etc. in a live Azure environment. (All environments have a functioning WAF).
- Report any WAF complications to the DevOps team as soon as you can, so that a mitigation can be included before your changes go to Production.
- If a mitigation is needed, please refrain from merging your changes until the mitigation is included in your branch.
- Regularly audit the WAF logs for signs that additional tuning or mitigations may be needed.
- Quickly respond to Engineering and Support requests for tuning and mitigations
- Proactively advise on the need for new mitigations as new features are designed
- Use the diagnostic steps above when gathering information about a user encounter with the WAF
- Raise the issue to the engineering teams as soon as practicable for proper mitigation.
- Getting Started
- [Setup] Docker and docker compose development
- [Setup] IntelliJ run configurations
- [Setup] Running DB outside of Docker (optional)
- [Setup] Running nginx locally (optional)
- [Setup] Running outside of docker
- Accessing and testing weird parts of the app on local dev
- Accessing patient experience in local dev
- API Testing with Insomnia
- Cypress
- How to run e2e locally for development
- E2E tests
- Database maintenance
- MailHog
- Running tests
- SendGrid
- Setting up okta
- Sonar
- Storybook and Chromatic
- Twilio
- User roles
- Wiremock
- CSV Uploader
- Log local DB queries
- Code review and PR conventions
- SimpleReport Style Guide
- How to Review and Test Pull Requests for Dependabot
- How to Review and Test Pull Requests with Terraform Changes
- SimpleReport Deployment Process
- Adding a Developer
- Removing a developer
- Non-deterministic test tracker
- Alert Response - When You Know What is Wrong
- What to Do When You Have No Idea What is Wrong
- Main Branch Status
- Maintenance Mode
- Swapping Slots
- Monitoring
- Container Debugging
- Debugging the ReportStream Uploader
- Renew Azure Service Principal Credentials
- Releasing Changelog Locks
- Muting Alerts
- Architectural Decision Records
- Backend Stack Overview
- Frontend Overview
- Cloud Architecture
- Cloud Environments
- Database ERD
- External IDs
- GraphQL Flow
- Hibernate Lazy fetching and nested models
- Identity Verification (Experian)
- Spring Profile Management
- SR Result bulk uploader device validation logic
- Test Metadata and how we store it
- TestOrder vs TestEvent
- ReportStream Integration
- Feature Flag Setup
- FHIR Resources
- FHIR Conversions
- Okta E2E Integration
- Deploy Application Action
- Slack notifications for support escalations
- Creating a New Environment Within a Resource Group
- How to Add and Use Environment Variables in Azure
- Web Application Firewall (WAF) Troubleshooting and Maintenance
- How to Review and Test Pull Requests with Terraform Changes