-
-
Notifications
You must be signed in to change notification settings - Fork 288
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
Fix RCE in gateway.downloadBackup #2171
base: master
Are you sure you want to change the base?
Conversation
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## master #2171 +/- ##
==========================================
+ Coverage 98.52% 98.53% +0.01%
==========================================
Files 868 876 +8
Lines 14295 14402 +107
==========================================
+ Hits 14084 14191 +107
Misses 211 211 ☔ View full report in Codecov by Sentry. 🚨 Try these New Features:
|
#2927 Bundle Size — 10.41MiB (0%).0a4a46c(current) vs f961aef master#2919(baseline) Warning Bundle contains 3 duplicate packages – View duplicate packages Bundle metrics
|
Current #2927 |
Baseline #2919 |
|
---|---|---|
Initial JS | 5.64MiB |
5.64MiB |
Initial CSS | 304.89KiB |
304.89KiB |
Cache Invalidation | 0% |
0% |
Chunks | 51 |
51 |
Assets | 173 |
173 |
Modules | 1511 |
1511 |
Duplicate Modules | 21 |
21 |
Duplicate Code | 0.83% |
0.83% |
Packages | 124 |
124 |
Duplicate Packages | 3 |
3 |
Bundle size by type no changes
Current #2927 |
Baseline #2919 |
|
---|---|---|
JS | 7.43MiB |
7.43MiB |
IMG | 2.54MiB |
2.54MiB |
CSS | 321.79KiB |
321.79KiB |
Fonts | 93.55KiB |
93.55KiB |
Other | 17.79KiB |
17.79KiB |
HTML | 13.58KiB |
13.58KiB |
Bundle analysis report Branch bnematzadeh:gladys-sec-2 Project dashboard
Generated by RelativeCI Documentation Report issue
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for this PR! I'm not sure this could ever be exploited as the user is hosting the server himself, but it's still a good thing to fix this :)
Why not use an URL validator to make sure the URL is really valid?
We use Joi in Gladys:
const Joi = require('joi');
const schema = Joi.string().uri();
// Example validation
const url = 'https://example.com';
const { error, value } = schema.validate(url);
Don't forget to add the tests :)
This scenario can be exploited by a new user created by an admin by calling the mentioned endpoint. Using the JoI library you referred to, I defined a regex with the pattern method to remove the specified characters if they appear at the end of a URL and return the cleaned URL. This is because simply using the uri method in this case is not sufficient and can be bypassed with other payloads. I made some changes to coreErrors and added a new custom error for URLs, which I have used both in the gateway file and in its corresponding test file. |
Oops, I think we have a problem. With the changes I've made to the downloadBackup file, two other tests are failing because when the downloadBackup method is called with a non-URL input, it reaches the validator and causes an error. I made a change in the file where, using a function, it checks whether the input is a URL or not. If it is a URL, the validateUrl function is called on it, and if the URL is valid, the value is returned. Otherwise, we're dealing with a local file. |
@bnematzadeh I tried with Joi with the native uri check and it works: Why not use this one? Do you have example of security issues with this ? |
The reason I didn’t use the native URI is that it recognizes the following URL as a valid URL and doesn’t handle the rest of the fragment. A custom command can be used inside |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok thanks for you answer! I have a few feedbacks on the PR
// Extract file name | ||
const fileWithoutSignedParams = fileUrl.split('?')[0]; | ||
|
||
const value = isURL(fileUrl) ? validateUrl(fileUrl) : fileUrl; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure it still works like that, for context, the fileUrl looks like that (it's a dummy URL) :
https://example_storage.com/b6131425-8f90-4a05-ba00-12daf37f8279.enc?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=SDFDSFDSFDSFDSFFDF%2F20241202%2Fus-east-1%2Fs3%2Faws4_request&X-Amz-Date=20241202T110928Z&X-Amz-Expires=21600&X-Amz-Signature=6768767678HHkjhjdhfjdshfjhdksfhjdshf&X-Amz-SignedHeaders=host
So the get parameters should be removed before doing path.basename()
Having the variable being name value
isn't very clear
* @example | ||
* validateUrl(); | ||
*/ | ||
function validateUrl(url) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You created a generic function "validateUrl" in utils
, but isn't this behavior very gateway.downloadBackup specific?
Will this be used elsewhere?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What change should I make so that the GET parameter is removed before it’s passed to path.basename()? The validator I wrote, in my opinion, can be used in other places, but it’s largely tied to gateway.downloadBackup.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Using fileWithoutSignedParams
instead ? (as it was before)
const fileWithoutSignedParams = fileUrl.split('?')[0];
Description
There is a Remote Code Execution vulnerability in the gateway controller that allows a logged-in user to execute arbitrary commands on the service. The vulnerable sink is located in the
gateway.downloadBackup
file. ThefileUrl
value corresponds to the file_url field in the/api/v1/gateway/backup/restore
endpoint.In the gateway.downloadBackup file, the
GLADYS_GATEWAY_BACKUP_KEY
value is first checked, and if it is not set, an error is thrown (this can be triggered by calling thePOST /api/v1/variable/GLADYS_GATEWAY_BACKUP_KEY
endpoint). To decrypt the backup, theexec
function is used. However, the value of encryptedBackupFilePath is used in exec without being sanitized.Before executing exec, the server sends a request to the
fileUrl
. If the link is problematic, an Axios error is returned. If a query string is included in the URL, its value is ignored. The issue here is that a malicious link containing an encrypted file and a hash value with an arbitrary command can be crafted. When this hash value is used in exec, the command gets executed.Below, I have attached a sample POC and the command I executed. To fix this issue, the # character must also be stripped from the fileWithoutSignedParams.