fix(codgen-fetch): fix the broken codegen for multipart requests as well as refactor it

This commit is contained in:
Tanish2002
2024-03-14 02:36:06 +05:30
parent 897160f388
commit c37fed8997

View File

@ -1,4 +1,3 @@
import 'dart:convert';
import 'package:jinja/jinja.dart' as jj; import 'package:jinja/jinja.dart' as jj;
import 'package:apidash/utils/utils.dart' import 'package:apidash/utils/utils.dart'
show padMultilineString, requestModelToHARJsonRequest; show padMultilineString, requestModelToHARJsonRequest;
@ -11,14 +10,16 @@ class FetchCodeGen {
final bool isNodeJs; final bool isNodeJs;
String kStringImportNode = """ String kStringImportNode = """
import fetch from 'node-fetch'; import fetch from 'node-fetch'
{% if hasFormData %}const fs = require('fs');{% endif %} {% if hasFormData -%}
import { fileFromSync, FormData } from 'node-fetch'
{% endif %}
"""; """;
String kTemplateStart = """let url = '{{url}}'; String kTemplateStart = """const url = '{{url}}';
let options = { const options = {
method: '{{method}}' method: '{{method}}'
"""; """;
@ -27,72 +28,62 @@ let options = {
"""; """;
String kTemplateBody = """, String kTemplateBody = """,
body: body: {{body}}
{{body}}
"""; """;
String kMultiPartBodyTemplate = r''' String kMultiPartBodyTemplate = r'''
async function buildDataList(fields) { payload.append("{{name}}", {{value}})
var formdata = new FormData();
for (const field of fields) {
const name = field.name || '';
const value = field.value || '';
const type = field.type || 'text';
if (type === 'text') {
formdata.append(name, value);
} else if (type === 'file') {
formdata.append(name,{% if isNodeJs %} fs.createReadStream(value){% else %} fileInput.files[0],value{% endif %});
}
}
return formdata;
}
const payload = buildDataList({{fields_list}});
'''; ''';
String kStringRequest = """ String kStringRequest = """
}; };
let status;
fetch(url, options) fetch(url, options)
.then(res => { .then(res => {
status = res.status; console.log(res.status);
return res.json() return res.text()
}) })
.then(body => { .then(body => {
console.log(status);
console.log(body); console.log(body);
}) })
.catch(err => { .catch(err => {
console.log(status); console.error(`error:\${err}`);
console.error('error:' + err);
}); });
"""; """;
String? getCode( String? getCode(RequestModel requestModel) {
RequestModel requestModel,
) {
try { try {
jj.Template kNodejsImportTemplate = jj.Template(kStringImportNode); jj.Template kNodejsImportTemplate = jj.Template(kStringImportNode);
String importsData = kNodejsImportTemplate.render({ String importsData = kNodejsImportTemplate.render({
"hasFormData": requestModel.hasFormData, "hasFormData": requestModel.hasFormData,
}); });
String result = isNodeJs ? importsData : ""; String result = isNodeJs
? importsData
: requestModel.hasFormData
? "// refer https://github.com/foss42/apidash/issues/293#issuecomment-1995208098 for details regarding integration\n\n"
: "";
if (requestModel.hasFormData) { if (requestModel.hasFormData) {
result += "const payload = new FormData();\n";
var templateMultiPartBody = jj.Template(kMultiPartBodyTemplate); var templateMultiPartBody = jj.Template(kMultiPartBodyTemplate);
var formFileCounter = 1;
for (var element in requestModel.formDataMapList) {
result += templateMultiPartBody.render({ result += templateMultiPartBody.render({
"isNodeJs": isNodeJs, "name": element["name"],
"fields_list": json.encode(requestModel.formDataMapList), "value": element["type"] == "text"
? "\"${element["value"]}\""
: isNodeJs
? "fileFromSync(\"${element["value"]}\")"
: "fileInput$formFileCounter.files[0]"
}); });
if (element["type"] != "text") formFileCounter++;
}
result += "\n";
} }
var harJson = requestModelToHARJsonRequest( var harJson =
requestModel, requestModelToHARJsonRequest(requestModel, useEnabled: true);
useEnabled: true,
);
var templateStart = jj.Template(kTemplateStart); var templateStart = jj.Template(kTemplateStart);
result += templateStart.render({ result += templateStart.render({
@ -105,16 +96,19 @@ fetch(url, options)
if (headers.isNotEmpty) { if (headers.isNotEmpty) {
var templateHeader = jj.Template(kTemplateHeader); var templateHeader = jj.Template(kTemplateHeader);
var m = {}; var m = {};
if (requestModel.hasFormData) {
m[kHeaderContentType] = "multipart/form-data";
}
for (var i in headers) { for (var i in headers) {
// fetch can automatically add the Content-Type header when FormData is passed as body
if (i["name"] == "Content-Type" && requestModel.hasFormData) {
continue;
}
m[i["name"]] = i["value"]; m[i["name"]] = i["value"];
} }
if (m.isNotEmpty) {
result += templateHeader.render({ result += templateHeader.render({
"headers": padMultilineString(kEncoder.convert(m), 2), "headers": padMultilineString(kEncoder.convert(m), 2),
}); });
} }
}
if (harJson["postData"]?["text"] != null) { if (harJson["postData"]?["text"] != null) {
var templateBody = jj.Template(kTemplateBody); var templateBody = jj.Template(kTemplateBody);