6K Star 11.8K Fork 4K

GVPdotNET China / Furion

 / 详情

Swagger 增加复制功能

已完成
创建于  
2022-10-14 09:58

💢 特别说明:如果 Issue 没有严格按照模板编写且未提供测试源码下载或 Git 测试仓库地址,则视为无效 Issue,将无法得到答复。

Furion 版本号

哪个版本号?
最新版

.NET SDK 版本号

  • .NET5
  • .NET6
  • .NET7

Web 项目类型

  • WebApi
  • Mvc
  • Razor Pages
  • Blazor Server
  • MinApp

操作系统和版本

  • Windows(版本)
  • Linux(版本)
  • MacOS(版本)
  • 其他(版本)

代码环境

  • 开发环境(Development)
  • 生产环境(Production)
  • 测试环境(Tests/单元测试/集成测试 )

描述你的问题

发生了什么?
默认是没有复制功能;但是在swagger ui官方是可以的
https://petstore.swagger.io/?_ga=2.58441386.1495808074.1665711102-76860923.1665711102
输入图片说明
根据上面的css和js 修改furion 源代码 index-mini-profiler.html文件

<!-- HTML for static distribution bundle build -->
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>%(DocumentTitle)</title>
    <link rel="stylesheet" type="text/css" href="https://petstore.swagger.io/swagger-ui.css">
    <link rel="stylesheet" type="text/css" href="https://petstore.swagger.io/index.css">
    <link rel="icon" type="image/png" href="./favicon-32x32.png" sizes="32x32" />
    <link rel="icon" type="image/png" href="./favicon-16x16.png" sizes="16x16" />
    <style>

        html {
            box-sizing: border-box;
            overflow: -moz-scrollbars-vertical;
            overflow-y: scroll;
        }

        *,
        *:before,
        *:after {
            box-sizing: inherit;
        }

        body {
            margin: 0;
            background: #fafafa;
        }

        #login-form {
            position: fixed;
            top: 30px;
            left: 50%;
            margin-left: -191px;
            width: 382px;
            display: none;
            border-color: #49cc90;
            background: rgba(73,204,144,.1);
            border: 1px solid #000;
            border-radius: 4px;
            box-shadow: 0 0 3px rgb(0 0 0 / 19%);
            font-family: monospace;
            overflow: hidden;
        }

        .login-title {
            background-color: rgb(27, 27, 27);
            height: 50px;
            padding: 7px 0 0 7px;
        }

            .login-title img {
                display: block;
                height: 36px;
            }

        .login-area {
            padding: 25px 40px 32px;
            background-color: rgb(232, 246, 240);
        }

        .form-row {
            padding: 6px 0;
        }

            .form-row label {
                font-size: 14px;
                flex: 1;
                margin: 0;
                font-family: sans-serif;
                color: #3b4151;
                font-weight: 700;
            }

            .form-row input {
                width: 100%;
                max-width: 340px;
                min-width: 100px;
                margin: 5px 0;
                padding: 8px 10px;
                border: 1px solid #d9d9d9;
                border-radius: 4px;
                background: #fff;
            }

            .form-row button {
                padding: 8px 40px;
                font-size: 14px;
                border: none;
                outline: none;
                font-weight: 700;
                cursor: pointer;
                border-radius: 3px;
                background-color: #4990e2;
                color: #fff;
                border-color: #4990e2;
                box-shadow: 0 0 5px rgb(0 0 0 / 30%);
                width: 100%;
                box-sizing: border-box;
            }

                .form-row button:hover {
                    opacity: 0.8;
                }

        #login-error {
            font-size: 14px;
            flex: 1;
            margin: 0;
            font-family: sans-serif;
            color: red;
            margin-top: 8px;
        }
    </style>
    %(HeadContent)
</head>

<body>
    <div id="swagger-ui"></div>
    <div id="login-form">
        <div class="login-title">
            <img src="" alt="" />
        </div>
        <div class="login-area">
            <div class="form-row">
                <label for="userName">User Name: </label>
                <input type="text" id="userName" name="userName" />
            </div>
            <div class="form-row">
                <label for="password">Password: </label>
                <input type="password" id="password" name="password" />
            </div>
            <div class="form-row">
                <label></label>
                <button id="submit">Submit</button>
            </div>
            <div id="login-error"></div>
        </div>
    </div>
    <script async id="mini-profiler" src="/index-mini-profiler/includes.min.js?v=4.2.22+4563a9e1ab" data-version="4.2.22+4563a9e1ab" data-path="/index-mini-profiler/" data-current-id="7a3d98bb-3968-41fb-8836-65f9923ee6eb" data-ids="7a3d98bb-3968-41fb-8836-65f9923ee6eb" data-position="Left" data-scheme="Light" data-authorized="true" data-max-traces="15" data-toggle-shortcut="Alt+P" data-trivial-milliseconds="2.0" data-ignored-duplicate-execute-types="Open,OpenAsync,Close,CloseAsync"></script>
    <!-- Workaround for https://github.com/swagger-api/swagger-editor/issues/1371 -->
    <script>
        if (window.navigator.userAgent.indexOf("Edge") > -1) {
            console.log("Removing native Edge fetch in favor of swagger-ui's polyfill")
            window.fetch = undefined;
        }
    </script>

    <script src="https://petstore.swagger.io/swagger-ui-bundle.js"></script>
    <script src="https://petstore.swagger.io/swagger-ui-standalone-preset.js"></script>
    <script>
        /* Source: https://gist.github.com/lamberta/3768814
         * Parse a string function definition and return a function object. Does not use eval.
         * @param {string} str
         * @return {function}
         *
         * Example:
         *  var f = function (x, y) { return x * y; };
         *  var g = parseFunction(f.toString());
         *  g(33, 3); //=> 99
         */
        function parseFunction(str) {
            if (!str) return void (0);

            var fn_body_idx = str.indexOf('{'),
                fn_body = str.substring(fn_body_idx + 1, str.lastIndexOf('}')),
                fn_declare = str.substring(0, fn_body_idx),
                fn_params = fn_declare.substring(fn_declare.indexOf('(') + 1, fn_declare.lastIndexOf(')')),
                args = fn_params.split(',');

            args.push(fn_body);

            function Fn() {
                return Function.apply(this, args);
            }
            Fn.prototype = Function.prototype;

            return new Fn();
        }

        // get uri all query variables
        function getQueryVariables(uri) {
            var searchArgs = window.location.search;
            if (uri) {
                var s = uri.indexOf('?');
                searchArgs = s > -1 ? uri.substring(s) : '';
            }
            var query = searchArgs.substring(1);
            var vars = query.split('&');
            var varObj = {};
            for (var i = 0; i < vars.length; i++) {
                var pair = vars[i].split('=');
                varObj[pair[0]] = pair[1];
            }
            return varObj;
        }

        // get uri query variable
        function getQueryVariable(variable, uri) {
            var vars = getQueryVariables(uri);
            for (var key in vars) {
                if (key === variable) return vars[key];
            }
            return (false);
        }

        // if value is null or undefined than execute delete operate, else execute update
        function updateQueryVariable(uri, key, value) {
            var re = new RegExp('([?&])' + key + '=.*?(&|$)', 'i');
            var separator = uri.indexOf('?') !== -1 ? '&' : '?';
            if (uri.match(re)) {
                return uri.replace(re, !value ? '' : ('$1' + key + '=' + value + '$2'));
            }
            else {
                return !value ? '' : (uri + separator + key + '=' + value);
            }
        }

        // get uri template
        function getUrlTemplate(uri) {
            var reg = /\{(.+?)\}/g;
            return uri.match(reg);
        }

        // hide empty tags
        var HideEmptyTagsPlugin = function HideEmptyTagsPlugin() {
            return {
                statePlugins: {
                    spec: {
                        wrapSelectors: {
                            taggedOperations: function taggedOperations(ori) {
                                return function () {
                                    return ori.apply(void 0, arguments).filter(function (tagMeta) {
                                        return tagMeta.get("operations") && tagMeta.get("operations").size > 0;
                                    });
                                };
                            }
                        }
                    }
                }
            };
        };

        var tokenKey = "access-token";
        var cultureKey = "culture";

        // default response interceptor
        function defaultResponseInterceptor(response) {
            // handle jwt token
            var accessToken = response.headers[tokenKey];
            if (accessToken && accessToken != "invalid_token") {
                window.localStorage.setItem(tokenKey, accessToken);
                ui.preauthorizeApiKey("Bearer", accessToken);
            }
            else if (accessToken == "invalid_token") {
                window.localStorage.removeItem(tokenKey);
            }

            return response;
        }

        // default request interceptor
        function defaultRequestInterceptor(request) {
            var url = request.url;

            // add swagger header
            request.headers["request-from"] = "swagger";

            // handle template
            var temps = getUrlTemplate(url);
            if (temps && temps.length > 0) {
                for (var i = 0; i < temps.length; i++) {
                    var temp = temps[i];
                    var key = temp.substring(1, temp.length - 1);
                    var queryKey = getQueryVariable(key, url);
                    if (queryKey) {
                        url = updateQueryVariable(url.replace(temp, queryKey), key);
                    }
                    else url = url.replace(temp, '');
                }
                request.url = url;
            }

            // handle culture lang
            var culture = getQueryVariable(cultureKey);
            if (culture && culture.length > 0) {
                request.url = updateQueryVariable(url, cultureKey, culture);
            }

            return request;
        }

        // send ajax request
        function sendRequest(params) {
            let xhr = new XMLHttpRequest();
            xhr.onreadystatechange = function () {
                if (xhr.readyState == 4 && xhr.status == 200) {
                    let obj = JSON.parse(xhr.response);
                    params.success(obj);
                } else if (xhr.readyState == 4 && xhr.status >= 400) {
                    params.error(xhr);
                }
            }

            function objectToString(obj) {
                let arr = [];
                for (var k in obj) {
                    arr.push(`${k}=${obj[k]}`);
                }

                return arr.join('&');
            }

            let qs = objectToString(params.data);

            if (params.method.toUpperCase() == 'GET') {
                xhr.open('GET', params.url + '?' + qs);
                xhr.send();
            } else if (params.method.toUpperCase() == 'POST') {
                xhr.open('POST', params.url);
                xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
                xhr.send(qs);
            }
        }

        // init swagger ui
        function initSwaggerUI(configObject, oauthConfigObject) {
            configObject.onComplete = function () {
                var accessToken = window.localStorage.getItem(tokenKey);
                if (accessToken) {
                    ui.preauthorizeApiKey("Bearer", accessToken);
                }
            };

            // Workaround for https://github.com/swagger-api/swagger-ui/issues/5945
            configObject.urls.forEach(function (item) {
                if (item.url.startsWith("http") || item.url.startsWith("/")) return;
                item.url = window.location.href.replace("index.html", item.url).split('#')[0];
            });

            // If validatorUrl is not explicitly provided, disable the feature by setting to null
            if (!configObject.hasOwnProperty("validatorUrl"))
                configObject.validatorUrl = null

            // If oauth2RedirectUrl isn't specified, use the built-in default
            if (!configObject.hasOwnProperty("oauth2RedirectUrl"))
                configObject.oauth2RedirectUrl = (new URL("oauth2-redirect.html", window.location.href)).href;

            // Apply mandatory parameters
            configObject.dom_id = "#swagger-ui";
            configObject.presets = [SwaggerUIBundle.presets.apis, SwaggerUIStandalonePreset, HideEmptyTagsPlugin];
            configObject.layout = "StandaloneLayout";

            // Parse and add interceptor functions
            var interceptors = JSON.parse('%(Interceptors)');
            if (interceptors.RequestInterceptorFunction)
                configObject.requestInterceptor = parseFunction(interceptors.RequestInterceptorFunction);
            if (interceptors.ResponseInterceptorFunction)
                configObject.responseInterceptor = parseFunction(interceptors.ResponseInterceptorFunction);

            // Begin Swagger UI call region

            const ui = SwaggerUIBundle(configObject);

            ui.initOAuth(oauthConfigObject);

            // End Swagger UI call region

            window.ui = ui
        }

        window.onload = function () {
            var configObject = JSON.parse('%(ConfigObject)');
            var oauthConfigObject = JSON.parse('%(OAuthConfigObject)');

            var loginObject = configObject.LoginInfo;
            if (loginObject && loginObject.enabled === true) {
                var loginForm = document.getElementById("login-form");
                var userName = document.getElementById("userName");
                var password = document.getElementById("password");
                var submit = document.getElementById("submit");
                var loginError = document.getElementById("login-error");
                loginForm.style.display = "block";

                sendRequest({
                    method: "POST",
                    url: loginObject.checkUrl,
                    success: function (res) {
                        if (res.toString() === "200") {
                            loginForm.style.display = "none";
                            initSwaggerUI(configObject, oauthConfigObject);
                        }
                        else {
                            userName.focus();

                            submit.addEventListener("click", function (ev) {
                                if (userName.value.trim().length === 0) {
                                    userName.focus();
                                    return;
                                }
                                if (password.value.trim().length === 0) {
                                    password.focus();
                                    return;
                                }

                                loginError.innerHTML = "";

                                sendRequest({
                                    method: "POST",
                                    url: loginObject.submitUrl,
                                    data: {
                                        userName: userName.value.trim(),
                                        password: password.value.trim()
                                    },
                                    success: function (res) {
                                        if (res.toString() === "200") {
                                            loginForm.style.display = "none";
                                            initSwaggerUI(configObject, oauthConfigObject);
                                        }
                                        else {
                                            loginError.innerHTML = "Invalid UserName or Password.";
                                        }
                                    },
                                    error: function (xhr) {
                                        if (xhr.status === 404) {
                                            loginError.innerHTML = "Not Found: " + loginObject.submitUrl;
                                        }
                                        else {
                                            loginError.innerHTML = "Internal ServerError: " + loginObject.submitUrl;
                                        }
                                    }
                                });
                            });
                        }
                    },
                    error: function (xhr) {
                        if (xhr.status === 404) {
                            loginError.innerHTML = "Not Found: " + loginObject.checkUrl;
                        }
                        else {
                            loginError.innerHTML = "Internal ServerError: " + loginObject.checkUrl;
                        }
                    }
                });
            }
            else {
                initSwaggerUI(configObject, oauthConfigObject);
            }
        }
    </script>
</body>
</html>

这样是可以的

输入图片说明

异常堆栈信息

异常堆栈是什么?


测试项目代码

⚠⚠ 必须提供完整可运行且包含错误的 Git 仓库 DEMO,DEMO 提供最简单的错误逻辑代码,否则将无法得到答复。⚠⚠

您的代码下载地址?


数据库信息

  • Sqlite
  • SqlServer
  • Mysql
  • Oracle
  • PGSql
  • Firebird
  • Cosmos

期待结果

期待的结果是?


评论 (7)

rockn 创建了任务

有没有更好的方法可以在代码中,开启或者禁用掉复制功能。

  1. Furion 集成的是第三方提供的 Swashbuckle.AspNetCore 包,这个包如何配置,那么你就如何配置,Furion 不做干涉。

  2. 你发的官方的 swagger 最新版 ui 未必和现在的 `Swashbuckle.AspNetCore 包兼容,就一个简单的例子,比如官方的 没有分组的功能


swagger 的东西,自行根据 Swashbuckle.AspNetCore 文档自由拓展。

百小僧 任务状态待办的 修改为已完成

久等了,今天正式支持 Swagger 路由地址 复制 URL 功能,将在 v4.8.4.13 版本发布

输入图片说明

Furion v4.8.4.13 版本发布,已经支持 Swagger 复制路由地址功能,如果更新后没看到,可以清空浏览器缓存查看

输入图片说明没有这个功能的时候我自定义了一个 :joy:

没事~~,升级至新版本就可以啦,省不少代码,哈哈。

登录 后才可以发表评论

状态
负责人
里程碑
Pull Requests
关联的 Pull Requests 被合并后可能会关闭此 issue
分支
开始日期   -   截止日期
-
置顶选项
优先级
参与者(3)
489708 rockn 1578925472 974299 monksoul 1578937227 5004810 fiying 1651212432
C#
1
https://gitee.com/dotnetchina/Furion.git
git@gitee.com:dotnetchina/Furion.git
dotnetchina
Furion
Furion

搜索帮助