PromucFlow_constructor/app/client/src/transformers/RestActionTransformer.ts
Valera Melnikov a2bfe450b6
chore: enable no-explicit-any rule (#35321)
## Description
-  Enabled the rule `@typescript-eslint/no-explicit-any`
- Suppressed errors with comment
```
// TODO: Fix this the next time the file is edited
// eslint-disable-next-line @typescript-eslint/no-explicit-any
```

Fixes #35308 

## Automation

/ok-to-test tags="@tag.All"

### 🔍 Cypress test results
<!-- This is an auto-generated comment: Cypress test results  -->
> [!TIP]
> 🟢 🟢 🟢 All cypress tests have passed! 🎉 🎉 🎉
> Workflow run:
<https://github.com/appsmithorg/appsmith/actions/runs/10181176984>
> Commit: 7fc604e24fa234da7ab2ff56e0b1c715268796ee
> <a
href="https://internal.appsmith.com/app/cypress-dashboard/rundetails-65890b3c81d7400d08fa9ee5?branch=master&workflowId=10181176984&attempt=2"
target="_blank">Cypress dashboard</a>.
> Tags: `@tag.All`
> Spec:
> <hr>Wed, 31 Jul 2024 15:00:45 UTC
<!-- end of auto-generated comment: Cypress test results  -->


## Communication
Should the DevRel and Marketing teams inform users about this change?
- [ ] Yes
- [x] No
2024-07-31 18:41:28 +03:00

142 lines
4.6 KiB
TypeScript

import {
HTTP_METHOD,
CONTENT_TYPE_HEADER_KEY,
} from "constants/ApiEditorConstants/CommonApiConstants";
import type { ApiAction } from "entities/Action";
import isEmpty from "lodash/isEmpty";
import isString from "lodash/isString";
import cloneDeep from "lodash/cloneDeep";
import {
getDynamicStringSegments,
isDynamicValue,
} from "utils/DynamicBindingUtils";
export const transformRestAction = (data: ApiAction): ApiAction => {
let action = cloneDeep(data);
const actionConfigurationHeaders = action.actionConfiguration.headers;
const actionConfigurationAutoGeneratedHeaders =
action?.actionConfiguration?.autoGeneratedHeaders || [];
const contentTypeHeaderIndex = actionConfigurationHeaders.findIndex(
(header: { key: string; value: string }) =>
header?.key?.trim().toLowerCase() === CONTENT_TYPE_HEADER_KEY,
);
const autoGeneratedContentTypeHeaderIndex =
actionConfigurationAutoGeneratedHeaders.findIndex(
(header: { key: string; value: string }) =>
header?.key?.trim().toLowerCase() === CONTENT_TYPE_HEADER_KEY,
);
// GET actions should not save body if the content-type is set to empty
// In all other scenarios, GET requests will save & execute the action with
// the request body
if (
action.actionConfiguration.httpMethod === HTTP_METHOD.GET &&
contentTypeHeaderIndex == -1 &&
autoGeneratedContentTypeHeaderIndex == -1
) {
delete action.actionConfiguration.body;
}
// Paths should not have query params
if (
action.actionConfiguration.queryParameters &&
action.actionConfiguration.queryParameters.length
) {
const path = action.actionConfiguration.path;
// This can help extract paths from the following examples
const templatePaths = extractApiUrlPath(path);
if (path && templatePaths !== path) {
action = {
...action,
actionConfiguration: {
...action.actionConfiguration,
path: templatePaths,
},
};
}
}
// Body should send correct format depending on the content type
if (action.actionConfiguration.httpMethod !== HTTP_METHOD.GET) {
// TODO: Fix this the next time the file is edited
// eslint-disable-next-line @typescript-eslint/no-explicit-any
let body: any = "";
if (action.actionConfiguration.body) {
body = action.actionConfiguration.body || undefined;
}
if (!isString(body)) body = JSON.stringify(body);
action = {
...action,
actionConfiguration: {
...action.actionConfiguration,
body,
},
};
}
action.actionConfiguration.bodyFormData = removeEmptyPairs(
action.actionConfiguration.bodyFormData,
);
action.actionConfiguration.headers = removeEmptyPairs(
action.actionConfiguration.headers,
);
action.actionConfiguration.queryParameters = removeEmptyPairs(
action.actionConfiguration.queryParameters,
);
return action;
};
// Filters empty key-value pairs or key-value-type(Multipart) from form data, headers and query params
// TODO: Fix this the next time the file is edited
// eslint-disable-next-line @typescript-eslint/no-explicit-any
function removeEmptyPairs(keyValueArray: any) {
if (!keyValueArray || !keyValueArray.length) return keyValueArray;
return keyValueArray.filter(
// TODO: Fix this the next time the file is edited
// eslint-disable-next-line @typescript-eslint/no-explicit-any
(data: any) =>
data &&
(!isEmpty(data.key) || !isEmpty(data.value) || !isEmpty(data.type)),
);
}
// This function extracts the appropriate paths regardless of whatever expressions exist within the dynamic bindings.
// Example 1: `/{{Text1.text ? 'users' : 'user'}}`
// Example 2: `/{{Text1.text ? 'users' : 'user'}}/{{"test"}}?`
// Example 3: `/{{Text1.text ? 'users' : 'user'}}/{{"test"}}?a=hello&b=world`
// Output 1: /{{Text1.text ? 'users' : 'user'}}`
// Output 2: /{{Text1.text ? 'users' : 'user'}}/{{"test"}}`
// Output 3: /{{Text1.text ? 'users' : 'user'}}/{{"test"}}`
export const extractApiUrlPath = (path: string | undefined) => {
const dynamicStringSegments = getDynamicStringSegments(path || "");
const dynamicValuesDetected: string[] = [];
const templateStringSegments = dynamicStringSegments.map((segment) => {
if (isDynamicValue(segment)) {
dynamicValuesDetected.push(segment);
return "~";
}
return segment;
});
const indexOfQueryParams = templateStringSegments.join("").indexOf("?");
let templatePaths = templateStringSegments
.join("")
.slice(0, indexOfQueryParams === -1 ? undefined : indexOfQueryParams);
dynamicValuesDetected.forEach((val) => {
templatePaths = templatePaths.replace(/~/, val);
});
return templatePaths;
};