parse-server 9.3.0-alpha.2 → 9.3.0-alpha.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +28 -0
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +24 -4
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +15 -7
- package/lib/Auth.js +16 -3
- package/lib/Config.js +12 -1
- package/lib/Controllers/DatabaseController.js +17 -1
- package/lib/Deprecator/Deprecations.js +6 -2
- package/lib/FileUrlValidator.js +69 -0
- package/lib/GraphQL/transformers/mutation.js +7 -1
- package/lib/Options/Definitions.js +7 -1
- package/lib/Options/docs.js +2 -1
- package/lib/Options/index.js +1 -1
- package/lib/ParseServer.js +2 -2
- package/lib/RestWrite.js +9 -2
- package/lib/Routers/FunctionsRouter.js +7 -1
- package/package.json +7 -7
- package/types/Options/index.d.ts +1 -0
package/README.md
CHANGED
|
@@ -68,6 +68,7 @@ A big _thank you_ 🙏 to our [sponsors](#sponsors) and [backers](#backers) who
|
|
|
68
68
|
- [Using Environment Variables](#using-environment-variables)
|
|
69
69
|
- [Available Adapters](#available-adapters)
|
|
70
70
|
- [Configuring File Adapters](#configuring-file-adapters)
|
|
71
|
+
- [Restricting File URL Domains](#restricting-file-url-domains)
|
|
71
72
|
- [Idempotency Enforcement](#idempotency-enforcement)
|
|
72
73
|
- [Localization](#localization)
|
|
73
74
|
- [Pages](#pages)
|
|
@@ -491,6 +492,33 @@ Parse Server allows developers to choose from several options when hosting files
|
|
|
491
492
|
|
|
492
493
|
`GridFSBucketAdapter` is used by default and requires no setup, but if you're interested in using Amazon S3, Google Cloud Storage, or local file storage, additional configuration information is available in the [Parse Server guide](http://docs.parseplatform.org/parse-server/guide/#configuring-file-adapters).
|
|
493
494
|
|
|
495
|
+
### Restricting File URL Domains
|
|
496
|
+
|
|
497
|
+
Parse objects can reference files by URL. To prevent [SSRF attacks](https://owasp.org/www-community/attacks/Server_Side_Request_Forgery) via crafted file URLs, you can restrict the allowed URL domains using the `fileUpload.allowedFileUrlDomains` option.
|
|
498
|
+
|
|
499
|
+
This protects against scenarios where an attacker provides a `Parse.File` with an arbitrary URL, for example as a Cloud Function parameter or in a field of type `Object` or `Array`. If Cloud Code or a client calls `getData()` on such a file, the Parse SDK makes an HTTP request to that URL, potentially leaking the server or client IP address and accessing internal services.
|
|
500
|
+
|
|
501
|
+
> [!NOTE]
|
|
502
|
+
> Fields of type `Parse.File` in the Parse schema are not affected by this attack, because Parse Server discards the URL on write and dynamically generates it on read based on the file adapter configuration.
|
|
503
|
+
|
|
504
|
+
```javascript
|
|
505
|
+
const parseServer = new ParseServer({
|
|
506
|
+
...otherOptions,
|
|
507
|
+
fileUpload: {
|
|
508
|
+
allowedFileUrlDomains: ['cdn.example.com', '*.example.com'],
|
|
509
|
+
},
|
|
510
|
+
});
|
|
511
|
+
```
|
|
512
|
+
|
|
513
|
+
| Parameter | Optional | Type | Default | Environment Variable |
|
|
514
|
+
|---|---|---|---|---|
|
|
515
|
+
| `fileUpload.allowedFileUrlDomains` | yes | `String[]` | `['*']` | `PARSE_SERVER_FILE_UPLOAD_ALLOWED_FILE_URL_DOMAINS` |
|
|
516
|
+
|
|
517
|
+
- `['*']` (default) allows file URLs with any domain.
|
|
518
|
+
- `['cdn.example.com']` allows only exact hostname matches.
|
|
519
|
+
- `['*.example.com']` allows any subdomain of `example.com`.
|
|
520
|
+
- `[]` blocks all file URLs; only files referenced by name are allowed.
|
|
521
|
+
|
|
494
522
|
## Idempotency Enforcement
|
|
495
523
|
|
|
496
524
|
**Caution, this is an experimental feature that may not be appropriate for production.**
|
|
@@ -62,25 +62,45 @@ class BaseAuthCodeAdapter extends _AuthAdapter.default {
|
|
|
62
62
|
// abstract method
|
|
63
63
|
throw new Error('getAccessTokenFromCode is not implemented');
|
|
64
64
|
}
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Validates auth data on login. In the standard auth flows (login, signup,
|
|
68
|
+
* update), `beforeFind` runs first and validates credentials, so no
|
|
69
|
+
* additional credential check is needed here.
|
|
70
|
+
*/
|
|
65
71
|
validateLogin(authData) {
|
|
66
|
-
// User validation is already done in beforeFind
|
|
67
72
|
return {
|
|
68
73
|
id: authData.id
|
|
69
74
|
};
|
|
70
75
|
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Validates auth data on first setup or when linking a new provider.
|
|
79
|
+
* In the standard auth flows, `beforeFind` runs first and validates
|
|
80
|
+
* credentials, so no additional credential check is needed here.
|
|
81
|
+
*/
|
|
71
82
|
validateSetUp(authData) {
|
|
72
|
-
// User validation is already done in beforeFind
|
|
73
83
|
return {
|
|
74
84
|
id: authData.id
|
|
75
85
|
};
|
|
76
86
|
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Returns the auth data to expose to the client after a query.
|
|
90
|
+
*/
|
|
77
91
|
afterFind(authData) {
|
|
78
92
|
return {
|
|
79
93
|
id: authData.id
|
|
80
94
|
};
|
|
81
95
|
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* Validates auth data on update. In the standard auth flows, `beforeFind`
|
|
99
|
+
* runs first for any changed auth data and validates credentials, so no
|
|
100
|
+
* additional credential check is needed here. Unchanged (echoed-back) data
|
|
101
|
+
* skips both `beforeFind` and validation entirely.
|
|
102
|
+
*/
|
|
82
103
|
validateUpdate(authData) {
|
|
83
|
-
// User validation is already done in beforeFind
|
|
84
104
|
return {
|
|
85
105
|
id: authData.id
|
|
86
106
|
};
|
|
@@ -96,4 +116,4 @@ class BaseAuthCodeAdapter extends _AuthAdapter.default {
|
|
|
96
116
|
}
|
|
97
117
|
}
|
|
98
118
|
exports.default = BaseAuthCodeAdapter;
|
|
99
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
119
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_AuthAdapter","_interopRequireDefault","require","e","__esModule","default","BaseAuthCodeAdapter","AuthAdapter","constructor","adapterName","validateOptions","options","Error","enableInsecureAuth","clientId","clientSecret","beforeFind","authData","code","access_token","Parse","OBJECT_NOT_FOUND","user","getUserFromAccessToken","id","VALIDATION_ERROR","getAccessTokenFromCode","redirect_uri","validateLogin","validateSetUp","afterFind","validateUpdate","parseResponseData","data","startPos","indexOf","endPos","jsonData","substring","JSON","parse","exports"],"sources":["../../../src/Adapters/Auth/BaseCodeAuthAdapter.js"],"sourcesContent":["// abstract class for auth code adapters\nimport AuthAdapter from './AuthAdapter';\nexport default class BaseAuthCodeAdapter extends AuthAdapter {\n  constructor(adapterName) {\n    super();\n    this.adapterName = adapterName;\n  }\n  validateOptions(options) {\n\n    if (!options) {\n      throw new Error(`${this.adapterName} options are required.`);\n    }\n\n    this.enableInsecureAuth = options.enableInsecureAuth;\n    if (this.enableInsecureAuth) {\n      return;\n    }\n\n    this.clientId = options.clientId;\n    this.clientSecret = options.clientSecret;\n\n    if (!this.clientId) {\n      throw new Error(`${this.adapterName} clientId is required.`);\n    }\n\n    if (!this.clientSecret) {\n      throw new Error(`${this.adapterName} clientSecret is required.`);\n    }\n  }\n\n  async beforeFind(authData) {\n    if (this.enableInsecureAuth && !authData?.code) {\n      if (!authData?.access_token) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${this.adapterName} auth is invalid for this user.`);\n      }\n\n      const user = await this.getUserFromAccessToken(authData.access_token, authData);\n\n      if (user.id !== authData.id) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${this.adapterName} auth is invalid for this user.`);\n      }\n\n      return;\n    }\n\n    if (!authData?.code) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `${this.adapterName} code is required.`);\n    }\n\n    const access_token = await this.getAccessTokenFromCode(authData);\n    const user = await this.getUserFromAccessToken(access_token, authData);\n\n    if (authData.id && user.id !== authData.id) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${this.adapterName} auth is invalid for this user.`);\n    }\n\n    authData.access_token = access_token;\n    authData.id = user.id;\n\n    delete authData.code;\n    delete authData.redirect_uri;\n\n  }\n\n  async getUserFromAccessToken() {\n    // abstract method\n    throw new Error('getUserFromAccessToken is not implemented');\n  }\n\n  async getAccessTokenFromCode() {\n    // abstract method\n    throw new Error('getAccessTokenFromCode is not implemented');\n  }\n\n  /**\n   * Validates auth data on login. In the standard auth flows (login, signup,\n   * update), `beforeFind` runs first and validates credentials, so no\n   * additional credential check is needed here.\n   */\n  validateLogin(authData) {\n    return {\n      id: authData.id,\n    }\n  }\n\n  /**\n   * Validates auth data on first setup or when linking a new provider.\n   * In the standard auth flows, `beforeFind` runs first and validates\n   * credentials, so no additional credential check is needed here.\n   */\n  validateSetUp(authData) {\n    return {\n      id: authData.id,\n    }\n  }\n\n  /**\n   * Returns the auth data to expose to the client after a query.\n   */\n  afterFind(authData) {\n    return {\n      id: authData.id,\n    }\n  }\n\n  /**\n   * Validates auth data on update. In the standard auth flows, `beforeFind`\n   * runs first for any changed auth data and validates credentials, so no\n   * additional credential check is needed here. Unchanged (echoed-back) data\n   * skips both `beforeFind` and validation entirely.\n   */\n  validateUpdate(authData) {\n    return {\n      id: authData.id,\n    }\n  }\n\n  parseResponseData(data) {\n    const startPos = data.indexOf('(');\n    const endPos = data.indexOf(')');\n    if (startPos === -1 || endPos === -1) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${this.adapterName} auth is invalid for this user.`);\n    }\n    const jsonData = data.substring(startPos + 1, endPos);\n    return JSON.parse(jsonData);\n  }\n}\n"],"mappings":"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","ignoreList":[]}
|