apifox-openapi-transformer

transform apifox openapi format provided by `http://127.0.0.1:4523/export/openapi/` into typescript api flie.

  1. // ==UserScript==
  2. // @name apifox-openapi-transformer
  3. // @namespace https://github.com/zenonux
  4. // @version 1.5.1
  5. // @description transform apifox openapi format provided by `http://127.0.0.1:4523/export/openapi/` into typescript api flie.
  6. // @author 换个头像心好累
  7. // @license GPL-3.0 License
  8. // @match *://127.0.0.1:4523/export/openapi/*
  9. // @icon 
  10. // @grant GM_xmlhttpRequest
  11. // ==/UserScript==
  12.  
  13. (function () {
  14. "use strict";
  15.  
  16. window.onload = _init;
  17.  
  18. async function _init() {
  19. let dataString = await getApiDataString()
  20. const jsonData = JSON.parse(dataString);
  21. const apis = [];
  22. Object.keys(jsonData.paths).forEach((url) => {
  23. if (jsonData.paths[url].get) {
  24. apis.push(_buildApi(url, "get", jsonData.paths[url].get));
  25. }
  26. if (jsonData.paths[url].post) {
  27. apis.push(_buildApi(url, "post", jsonData.paths[url].post));
  28. }
  29. if (jsonData.paths[url].put) {
  30. apis.push(_buildApi(url, "put", jsonData.paths[url].put));
  31. }
  32. if (jsonData.paths[url].delete) {
  33. apis.push(_buildApi(url, "delete", jsonData.paths[url].delete));
  34. }
  35. });
  36. if (apis.length > 0) {
  37. let apiStr = apis.join("");
  38. _showExportButton(apiStr);
  39. }
  40. }
  41.  
  42.  
  43. function getApiDataString() {
  44. return new Promise((resolve, reject) => {
  45. GM_xmlhttpRequest({
  46. method: 'GET',
  47. url: window.location.href,
  48. onload: function (response) {
  49. if (response.status === 200) {
  50. resolve(response.responseText)
  51. } else {
  52. reject()
  53. }
  54. },
  55. onerror: function (error) {
  56. reject(error)
  57. }
  58. });
  59.  
  60. })
  61. }
  62.  
  63. function _buildApi(url, method, info) {
  64. let { summary, parameters, requestBody, responses } = info;
  65. let ParamsType = "{}";
  66. if (parameters && parameters.length > 0) {
  67. ParamsType = _parseParam(parameters);
  68. }
  69. let DataType = "{}";
  70. if (requestBody) {
  71. let body = requestBody.content["application/json"].schema;
  72. DataType = _parseBodyOrResponse(body.properties, body.required);
  73. }
  74. let ResponseType = "{}";
  75. let responseData = responses["200"].content["application/json"].schema;
  76. ResponseType = _parseBodyOrResponse(
  77. responseData.properties,
  78. responseData.required
  79. );
  80.  
  81. return `\n
  82. // ${summary}
  83. ${method}(payloads:{
  84. params${_isEmptyObjectStr(ParamsType) ? "?" : ""}:${ParamsType};
  85. data${_isEmptyObjectStr(DataType) ? "?" : ""}:${DataType};
  86. }): Promise<${ResponseType}>{
  87. return request({
  88. url:"${url}",
  89. method:"${method}",
  90. ${_isEmptyObjectStr(ParamsType) ? "" : "params:payloads.params,"}
  91. ${_isEmptyObjectStr(DataType) ? "" : "data:payloads.data,"}
  92. })
  93. },
  94. \n`;
  95. }
  96. function _parseParam(parameters) {
  97. let query = "{";
  98. parameters.forEach((v) => {
  99. // ignore in path
  100. if (v.in.indexOf("query") !== -1) {
  101. query += `${[v.name]}${v.required ? "" : "?"}:${v.schema.type};`;
  102. }
  103. });
  104. query += "}";
  105. return _replaceInteger2Number(query);
  106. }
  107. function _parseBodyOrResponse(properties, required = []) {
  108. if (!properties) {
  109. return '{}'
  110. }
  111. let data = "{";
  112. Object.keys(properties).forEach((v) => {
  113. let isRequired = required.some((k) => v === k);
  114. let type = properties[v].type;
  115. if (type === "object") {
  116. data += `${v}:`;
  117. data += _parseBodyOrResponse(
  118. properties[v].properties,
  119. properties[v].required
  120. );
  121. data += `;`;
  122. } else if (type === "array") {
  123. data += `${v}:`;
  124. data += _parseBodyOrResponse(
  125. properties[v].items.properties,
  126. properties[v].items.required
  127. );
  128. data += `[];`;
  129. } else {
  130. data += `${[v]}${isRequired ? "" : "?"}:${type};`;
  131. }
  132. });
  133. data += "}";
  134. return _replaceInteger2Number(data);
  135. }
  136. function _isEmptyObjectStr(obj) {
  137. return obj === "{}";
  138. }
  139. function _showExportButton(apiStr) {
  140. let btn = document.createElement("button");
  141. btn.innerText = "复制接口";
  142. btn.style =
  143. "position:fixed;top:16%;left:50%;transform:translate(-50%,0);z-index:10000;background:#888;padding:10px 14px;border:none;color:#fff;cursor:pointer;";
  144. btn.addEventListener("click", () => {
  145. _copyToClipboard(apiStr).then(() => {
  146. _showToast("复制成功");
  147. });
  148. });
  149. document.body.appendChild(btn);
  150. }
  151. function _replaceInteger2Number(text) {
  152. return text.replace(/integer/g, "number");
  153. }
  154. function _showToast(msg) {
  155. let toast = document.createElement("div");
  156. toast.innerText = msg;
  157. toast.style =
  158. "position:fixed;top:16%;left:50%;transform:translate(-50%,-24%);z-index:20000;background:#67C23A;padding:30px 45px;border-radius:4px;color:#fff;";
  159. document.body.appendChild(toast);
  160. setTimeout(() => {
  161. document.body.removeChild(toast);
  162. }, 1000);
  163. }
  164. function _copyToClipboard(text) {
  165. return navigator.clipboard.writeText(text);
  166. }
  167. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址