apifox-openapi-transformer

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

当前为 2023-04-08 提交的版本,查看 最新版本

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

QingJ © 2025

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