【AI】SpringCloudAlibaba AI 学习

news2024/11/23 0:10:53

Spring Cloud Alibaba AI 简介

Spring Cloud Alibaba AI 以 Spring AI 为基础,并在此基础上提供阿里云通义系列大模型全面适配,让用户在 5 分钟内开发基于通义大模型的 Java AI 应用。
官网:
https://sca.aliyun.com/
https://sca.aliyun.com/docs/2023/user-guide/ai/quick-start/

代码示例

代码仓库地址:https://gitee.com/wheelmouse/spring-ai-alibaba-demo.git
创建项目:spring-ai-alibaba-demo
pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>3.3.0</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <artifactId>spring-ai-alibaba-demo</artifactId>

    <properties>
        <maven.compiler.source>17</maven.compiler.source>
        <maven.compiler.target>17</maven.compiler.target>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-ai</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>com.alibaba.cloud</groupId>
                <artifactId>spring-cloud-alibaba-dependencies</artifactId>
                <version>2023.0.1.0</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <version>3.3.0</version>
                <configuration>
                    <excludes>
                        <exclude>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                        </exclude>
                    </excludes>
                </configuration>
            </plugin>
        </plugins>
    </build>

    <repositories>
        <repository>
            <id>spring-milestones</id>
            <name>Spring Milestones</name>
            <url>https://repo.spring.io/milestone</url>
            <snapshots>
                <enabled>false</enabled>
            </snapshots>
        </repository>
    </repositories>

</project>

写yml

server:
  port: 1352

spring:
  application:
    name: alibaba-spring-ai-demo
  cloud:
    ai:
      tongyi:
        api-key: # api-key的获取可以阅读下面 开通DashScope灵积模型服务 

启动类

@SpringBootApplication
public class SpringAiAlibabaApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringAiAlibabaApplication.class, args);
    }

}

Controller

package com.wheelmouse.ai.controller;

import com.wheelmouse.ai.service.TongYiService;
import jakarta.annotation.Resource;
import org.springframework.ai.image.ImageResponse;
import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

import java.util.Map;

/**
 * @author wheelmouse
 * @date 2024/7/1
 * @apiNote
 */
@CrossOrigin
@RestController
public class TongYiController {

    @Resource
    private TongYiService tongYiService;

    @GetMapping("/ai/simple")
    public String completion(@RequestParam("message") String message) {
        return tongYiService.completion(message);
    }

    @GetMapping("/ai/streamCompletion")
    public Map<String, String> streamCompletion(@RequestParam("message") String message) {
        return tongYiService.streamCompletion(message);
    }

    @GetMapping("/ai/genImg")
    public ImageResponse genImg(@RequestParam("imgPrompt") String imgPrompt) {
        return tongYiService.genImg(imgPrompt);
    }

}

Service及实现类

package com.wheelmouse.ai.service;

import org.springframework.ai.image.ImageResponse;

import java.util.Map;

/**
 * @author wheelmouse
 * @date 2024/7/1
 * @apiNote
 */
public interface TongYiService {
    /**
     * 基本问答
     */
    String completion(String message);

    /**
     * 流式基本问答
     */
    Map<String, String> streamCompletion(String message);

    /**
     * 文生图
     */
    ImageResponse genImg(String imgPrompt);

    /**
     * 语音合成
     */
    String genAudio(String text);
}

impl

package com.wheelmouse.ai.service.impl;

import com.wheelmouse.ai.service.TongYiService;
import jakarta.annotation.Resource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.ai.chat.ChatClient;
import org.springframework.ai.chat.StreamingChatClient;
import org.springframework.ai.chat.messages.UserMessage;
import org.springframework.ai.chat.prompt.Prompt;
import org.springframework.ai.image.ImageClient;
import org.springframework.ai.image.ImagePrompt;
import org.springframework.ai.image.ImageResponse;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import reactor.core.publisher.Flux;

import java.util.Map;

/**
 * @author wheelmouse
 * @date 2024/7/1
 * @apiNote
 */
@Service
public class TongYiSimpleServiceImpl implements TongYiService {

    private Logger log = LoggerFactory.getLogger(this.getClass());

    /**
     * 自动注入ChatClient、StreamingChatClient,屏蔽模型调用细节
     */
    private final ChatClient chatClient;
    private final StreamingChatClient streamingChatClient;

    @Autowired
    public TongYiSimpleServiceImpl(ChatClient chatClient, StreamingChatClient streamingChatClient) {
        this.chatClient = chatClient;
        this.streamingChatClient = streamingChatClient;
    }

    public String completion(String message) {
        Prompt prompt = new Prompt(new UserMessage(message));
        return chatClient.call(prompt).getResult().getOutput().getContent();
    }

    @Override
    public Map<String, String> streamCompletion(String message) {

        StringBuilder fullContent = new StringBuilder();

        streamingChatClient.stream(new Prompt(message))
                .flatMap(chatResponse -> Flux.fromIterable(chatResponse.getResults()))
                .map(content -> content.getOutput().getContent())
                .doOnNext(fullContent::append)
                .last()
                .map(lastContent -> Map.of(message, fullContent.toString()))
                .block();

        log.info(fullContent.toString());

        return Map.of(message, fullContent.toString());
    }

    @Resource
    private ImageClient imageClient;

    @Override
    public ImageResponse genImg(String imgPrompt) {
        var prompt = new ImagePrompt(imgPrompt);
        return imageClient.call(prompt);
    }

    @Override
    public String genAudio(String text) {
        return null;
    }
}

为了方便看效果,这里我参考了阿里官方案例制作了一个页面进行效果展示
在 resource 文件夹下创建 static/js/marked.min.js 和 static/index.html,2个文件代码如下:
index.html

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <script src="js/marked.min.js"></script>
    <title>SCA AI Example Front</title>
    <style>
        body {
            background-color: #f8f9fa;
            font-family: Arial, sans-serif;
        }

        .container {
            margin: 50px auto;
            width: 800px;
            background-color: #fff;
            padding: 20px;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
        }

        h1 {
            color: #2ecc71;
            text-align: center;
            margin-bottom: 30px;
        }

        label {
            display: block;
            margin-bottom: 10px;
            color: #333;
        }

        input[type="text"] {
            width: 100%;
            padding: 10px;
            border: 1px solid #ccc;
            border-radius: 3px;
        }

        input[type="submit"] {
            background-color: #2ecc71;
            color: #fff;
            border: none;
            padding: 10px 20px;
            border-radius: 3px;
            cursor: pointer;
            width: 100%;
        }

        .chat-box {
            width: 100%;
            height: 500px;
            padding: 10px;
            border: 1px solid #ccc;
            border-radius: 3px;
            overflow-y: scroll;
        }

        .message {
            margin-bottom: 10px;
            padding: 10px;
            background-color: #f1f1f1;
            border-radius: 3px;
        }

        .user-message {
            background-color: #2ecc71;
            color: #fff;
        }

        .bot-message {
            background-color: #3498db;
            color: #fff;
        }

        .loader {
            text-align: center;
        }

        .loader::after {
            content: "";
            display: inline-block;
            width: 20px;
            height: 20px;
            border-radius: 50%;
            border: 2px solid #ccc;
            border-top-color: #2ecc71;
            animation: spin 1s infinite linear;
        }

        @keyframes spin {
            to {
                transform: rotate(360deg);
            }
        }
    </style>
</head>
<body>
<div class="container">
    <h1>Spring Cloud Alibaba AI Example</h1>
    <form id="form">
        <label for="message">请输入你的问题:</label>
        <input type="text" id="message" name="message" placeholder="请输入你的问题!">
        <br>
        <br>
        <input type="submit" value="发送">
    </form>
    <br>
    <div id="loader" class="loader" style="display: none;"></div>
    <div id="chat-box" class="chat-box"></div>
</div>

<script>
    var loader = document.getElementById("loader");

    document.getElementById("form").addEventListener("submit", function (event) {
        event.preventDefault();

        var messageInput = document.getElementById("message");
        var message = messageInput.value;
        messageInput.value = "";

        var chatBox = document.getElementById("chat-box");

        var userMessage = document.createElement("div");
        userMessage.className = "message user-message";
        userMessage.textContent = "User: " + message;
        chatBox.appendChild(userMessage);
        chatBox.scrollTop = chatBox.scrollHeight;

        loader.style.display = "block";

        var xhr = new XMLHttpRequest();
        xhr.open("GET", "http://localhost:1352/ai/streamCompletion?message=" + encodeURIComponent(message), true);
        xhr.onreadystatechange = function () {
            if (xhr.readyState === 4) {
                loader.style.display = "none";

                if (xhr.status === 200) {
                    var response = xhr.responseText;

                    var botMessage = document.createElement("div");
                    botMessage.className = "message bot-message";

                    var botMessageText = document.createElement("span");
                    botMessageText.className = "message-text";
                    botMessage.appendChild(botMessageText);
                    botMessageText.innerHTML = marked.marked(response);

                    chatBox.appendChild(botMessage);
                    chatBox.scrollTop = chatBox.scrollHeight;
                } else if (xhr.status === 400) {
                    var error = JSON.parse(xhr.responseText);

                    var errorMessage = document.createElement("div");
                    errorMessage.className = "message bot-message";
                    errorMessage.textContent = "Bot: " + error.message;
                    chatBox.appendChild(errorMessage);
                    chatBox.scrollTop = chatBox.scrollHeight;
                } else {
                    var errorMessage = document.createElement("div");
                    errorMessage.className = "message bot-message";
                    errorMessage.textContent = "Bot: Failed to connect to the backend service. Please make sure the backend service is running.";
                    chatBox.appendChild(errorMessage);
                    chatBox.scrollTop = chatBox.scrollHeight;
                }
            }
        };

        xhr.onloadstart = function () {
            loader.style.display = "block";
        };

        xhr.onloadend = function () {
            loader.style.display = "none";
        };

        xhr.send();
    });
</script>
</body>
</html>

marked.min.js

/**
 * marked v12.0.1 - a markdown parser
 * Copyright (c) 2011-2024, Christopher Jeffrey. (MIT Licensed)
 * https://github.com/markedjs/marked
 */
!function (e, t) {
    "object" == typeof exports && "undefined" != typeof module ? t(exports) : "function" == typeof define && define.amd ? define(["exports"], t) : t((e = "undefined" != typeof globalThis ? globalThis : e || self).marked = {})
}(this, (function (e) {
        "use strict";

        function t() {
            return {
                async: !1,
                breaks: !1,
                extensions: null,
                gfm: !0,
                hooks: null,
                pedantic: !1,
                renderer: null,
                silent: !1,
                tokenizer: null,
                walkTokens: null
            }
        }

        function n(t) {
            e.defaults = t
        }

        e.defaults = {
            async: !1,
            breaks: !1,
            extensions: null,
            gfm: !0,
            hooks: null,
            pedantic: !1,
            renderer: null,
            silent: !1,
            tokenizer: null,
            walkTokens: null
        };
        const s = /[&<>"']/
            , r = new RegExp(s.source, "g")
            , i = /[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/
            , l = new RegExp(i.source, "g")
            , o = {
            "&": "&amp;",
            "<": "&lt;",
            ">": "&gt;",
            '"': "&quot;",
            "'": "&#39;"
        }
            , a = e => o[e];

        function c(e, t) {
            if (t) {
                if (s.test(e))
                    return e.replace(r, a)
            } else if (i.test(e))
                return e.replace(l, a);
            return e
        }

        const h = /&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/gi;

        function p(e) {
            return e.replace(h, ((e, t) => "colon" === (t = t.toLowerCase()) ? ":" : "#" === t.charAt(0) ? "x" === t.charAt(1) ? String.fromCharCode(parseInt(t.substring(2), 16)) : String.fromCharCode(+t.substring(1)) : ""))
        }

        const u = /(^|[^\[])\^/g;

        function k(e, t) {
            let n = "string" == typeof e ? e : e.source;
            t = t || "";
            const s = {
                replace: (e, t) => {
                    let r = "string" == typeof t ? t : t.source;
                    return r = r.replace(u, "$1"),
                        n = n.replace(e, r),
                        s
                }
                ,
                getRegex: () => new RegExp(n, t)
            };
            return s
        }

        function g(e) {
            try {
                e = encodeURI(e).replace(/%25/g, "%")
            } catch (e) {
                return null
            }
            return e
        }

        const f = {
            exec: () => null
        };

        function d(e, t) {
            const n = e.replace(/\|/g, ((e, t, n) => {
                    let s = !1
                        , r = t;
                    for (; --r >= 0 && "\\" === n[r];)
                        s = !s;
                    return s ? "|" : " |"
                }
            )).split(/ \|/);
            let s = 0;
            if (n[0].trim() || n.shift(),
            n.length > 0 && !n[n.length - 1].trim() && n.pop(),
                t)
                if (n.length > t)
                    n.splice(t);
                else
                    for (; n.length < t;)
                        n.push("");
            for (; s < n.length; s++)
                n[s] = n[s].trim().replace(/\\\|/g, "|");
            return n
        }

        function x(e, t, n) {
            const s = e.length;
            if (0 === s)
                return "";
            let r = 0;
            for (; r < s;) {
                const i = e.charAt(s - r - 1);
                if (i !== t || n) {
                    if (i === t || !n)
                        break;
                    r++
                } else
                    r++
            }
            return e.slice(0, s - r)
        }

        function b(e, t, n, s) {
            const r = t.href
                , i = t.title ? c(t.title) : null
                , l = e[1].replace(/\\([\[\]])/g, "$1");
            if ("!" !== e[0].charAt(0)) {
                s.state.inLink = !0;
                const e = {
                    type: "link",
                    raw: n,
                    href: r,
                    title: i,
                    text: l,
                    tokens: s.inlineTokens(l)
                };
                return s.state.inLink = !1,
                    e
            }
            return {
                type: "image",
                raw: n,
                href: r,
                title: i,
                text: c(l)
            }
        }

        class w {
            options;
            rules;
            lexer;

            constructor(t) {
                this.options = t || e.defaults
            }

            space(e) {
                const t = this.rules.block.newline.exec(e);
                if (t && t[0].length > 0)
                    return {
                        type: "space",
                        raw: t[0]
                    }
            }

            code(e) {
                const t = this.rules.block.code.exec(e);
                if (t) {
                    const e = t[0].replace(/^ {1,4}/gm, "");
                    return {
                        type: "code",
                        raw: t[0],
                        codeBlockStyle: "indented",
                        text: this.options.pedantic ? e : x(e, "\n")
                    }
                }
            }

            fences(e) {
                const t = this.rules.block.fences.exec(e);
                if (t) {
                    const e = t[0]
                        , n = function (e, t) {
                        const n = e.match(/^(\s+)(?:```)/);
                        if (null === n)
                            return t;
                        const s = n[1];
                        return t.split("\n").map((e => {
                                const t = e.match(/^\s+/);
                                if (null === t)
                                    return e;
                                const [n] = t;
                                return n.length >= s.length ? e.slice(s.length) : e
                            }
                        )).join("\n")
                    }(e, t[3] || "");
                    return {
                        type: "code",
                        raw: e,
                        lang: t[2] ? t[2].trim().replace(this.rules.inline.anyPunctuation, "$1") : t[2],
                        text: n
                    }
                }
            }

            heading(e) {
                const t = this.rules.block.heading.exec(e);
                if (t) {
                    let e = t[2].trim();
                    if (/#$/.test(e)) {
                        const t = x(e, "#");
                        this.options.pedantic ? e = t.trim() : t && !/ $/.test(t) || (e = t.trim())
                    }
                    return {
                        type: "heading",
                        raw: t[0],
                        depth: t[1].length,
                        text: e,
                        tokens: this.lexer.inline(e)
                    }
                }
            }

            hr(e) {
                const t = this.rules.block.hr.exec(e);
                if (t)
                    return {
                        type: "hr",
                        raw: t[0]
                    }
            }

            blockquote(e) {
                const t = this.rules.block.blockquote.exec(e);
                if (t) {
                    const e = x(t[0].replace(/^ *>[ \t]?/gm, ""), "\n")
                        , n = this.lexer.state.top;
                    this.lexer.state.top = !0;
                    const s = this.lexer.blockTokens(e);
                    return this.lexer.state.top = n,
                        {
                            type: "blockquote",
                            raw: t[0],
                            tokens: s,
                            text: e
                        }
                }
            }

            list(e) {
                let t = this.rules.block.list.exec(e);
                if (t) {
                    let n = t[1].trim();
                    const s = n.length > 1
                        , r = {
                        type: "list",
                        raw: "",
                        ordered: s,
                        start: s ? +n.slice(0, -1) : "",
                        loose: !1,
                        items: []
                    };
                    n = s ? `\\d{1,9}\\${n.slice(-1)}` : `\\${n}`,
                    this.options.pedantic && (n = s ? n : "[*+-]");
                    const i = new RegExp(`^( {0,3}${n})((?:[\t ][^\\n]*)?(?:\\n|$))`);
                    let l = ""
                        , o = ""
                        , a = !1;
                    for (; e;) {
                        let n = !1;
                        if (!(t = i.exec(e)))
                            break;
                        if (this.rules.block.hr.test(e))
                            break;
                        l = t[0],
                            e = e.substring(l.length);
                        let s = t[2].split("\n", 1)[0].replace(/^\t+/, (e => " ".repeat(3 * e.length)))
                            , c = e.split("\n", 1)[0]
                            , h = 0;
                        this.options.pedantic ? (h = 2,
                            o = s.trimStart()) : (h = t[2].search(/[^ ]/),
                            h = h > 4 ? 1 : h,
                            o = s.slice(h),
                            h += t[1].length);
                        let p = !1;
                        if (!s && /^ *$/.test(c) && (l += c + "\n",
                            e = e.substring(c.length + 1),
                            n = !0),
                            !n) {
                            const t = new RegExp(`^ {0,${Math.min(3, h - 1)}}(?:[*+-]|\\d{1,9}[.)])((?:[ \t][^\\n]*)?(?:\\n|$))`)
                                ,
                                n = new RegExp(`^ {0,${Math.min(3, h - 1)}}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)`)
                                , r = new RegExp(`^ {0,${Math.min(3, h - 1)}}(?:\`\`\`|~~~)`)
                                , i = new RegExp(`^ {0,${Math.min(3, h - 1)}}#`);
                            for (; e;) {
                                const a = e.split("\n", 1)[0];
                                if (c = a,
                                this.options.pedantic && (c = c.replace(/^ {1,4}(?=( {4})*[^ ])/g, "  ")),
                                    r.test(c))
                                    break;
                                if (i.test(c))
                                    break;
                                if (t.test(c))
                                    break;
                                if (n.test(e))
                                    break;
                                if (c.search(/[^ ]/) >= h || !c.trim())
                                    o += "\n" + c.slice(h);
                                else {
                                    if (p)
                                        break;
                                    if (s.search(/[^ ]/) >= 4)
                                        break;
                                    if (r.test(s))
                                        break;
                                    if (i.test(s))
                                        break;
                                    if (n.test(s))
                                        break;
                                    o += "\n" + c
                                }
                                p || c.trim() || (p = !0),
                                    l += a + "\n",
                                    e = e.substring(a.length + 1),
                                    s = c.slice(h)
                            }
                        }
                        r.loose || (a ? r.loose = !0 : /\n *\n *$/.test(l) && (a = !0));
                        let u, k = null;
                        this.options.gfm && (k = /^\[[ xX]\] /.exec(o),
                        k && (u = "[ ] " !== k[0],
                            o = o.replace(/^\[[ xX]\] +/, ""))),
                            r.items.push({
                                type: "list_item",
                                raw: l,
                                task: !!k,
                                checked: u,
                                loose: !1,
                                text: o,
                                tokens: []
                            }),
                            r.raw += l
                    }
                    r.items[r.items.length - 1].raw = l.trimEnd(),
                        r.items[r.items.length - 1].text = o.trimEnd(),
                        r.raw = r.raw.trimEnd();
                    for (let e = 0; e < r.items.length; e++)
                        if (this.lexer.state.top = !1,
                            r.items[e].tokens = this.lexer.blockTokens(r.items[e].text, []),
                            !r.loose) {
                            const t = r.items[e].tokens.filter((e => "space" === e.type))
                                , n = t.length > 0 && t.some((e => /\n.*\n/.test(e.raw)));
                            r.loose = n
                        }
                    if (r.loose)
                        for (let e = 0; e < r.items.length; e++)
                            r.items[e].loose = !0;
                    return r
                }
            }

            html(e) {
                const t = this.rules.block.html.exec(e);
                if (t) {
                    return {
                        type: "html",
                        block: !0,
                        raw: t[0],
                        pre: "pre" === t[1] || "script" === t[1] || "style" === t[1],
                        text: t[0]
                    }
                }
            }

            def(e) {
                const t = this.rules.block.def.exec(e);
                if (t) {
                    const e = t[1].toLowerCase().replace(/\s+/g, " ")
                        , n = t[2] ? t[2].replace(/^<(.*)>$/, "$1").replace(this.rules.inline.anyPunctuation, "$1") : ""
                        ,
                        s = t[3] ? t[3].substring(1, t[3].length - 1).replace(this.rules.inline.anyPunctuation, "$1") : t[3];
                    return {
                        type: "def",
                        tag: e,
                        raw: t[0],
                        href: n,
                        title: s
                    }
                }
            }

            table(e) {
                const t = this.rules.block.table.exec(e);
                if (!t)
                    return;
                if (!/[:|]/.test(t[2]))
                    return;
                const n = d(t[1])
                    , s = t[2].replace(/^\||\| *$/g, "").split("|")
                    , r = t[3] && t[3].trim() ? t[3].replace(/\n[ \t]*$/, "").split("\n") : []
                    , i = {
                    type: "table",
                    raw: t[0],
                    header: [],
                    align: [],
                    rows: []
                };
                if (n.length === s.length) {
                    for (const e of s)
                        /^ *-+: *$/.test(e) ? i.align.push("right") : /^ *:-+: *$/.test(e) ? i.align.push("center") : /^ *:-+ *$/.test(e) ? i.align.push("left") : i.align.push(null);
                    for (const e of n)
                        i.header.push({
                            text: e,
                            tokens: this.lexer.inline(e)
                        });
                    for (const e of r)
                        i.rows.push(d(e, i.header.length).map((e => ({
                            text: e,
                            tokens: this.lexer.inline(e)
                        }))));
                    return i
                }
            }

            lheading(e) {
                const t = this.rules.block.lheading.exec(e);
                if (t)
                    return {
                        type: "heading",
                        raw: t[0],
                        depth: "=" === t[2].charAt(0) ? 1 : 2,
                        text: t[1],
                        tokens: this.lexer.inline(t[1])
                    }
            }

            paragraph(e) {
                const t = this.rules.block.paragraph.exec(e);
                if (t) {
                    const e = "\n" === t[1].charAt(t[1].length - 1) ? t[1].slice(0, -1) : t[1];
                    return {
                        type: "paragraph",
                        raw: t[0],
                        text: e,
                        tokens: this.lexer.inline(e)
                    }
                }
            }

            text(e) {
                const t = this.rules.block.text.exec(e);
                if (t)
                    return {
                        type: "text",
                        raw: t[0],
                        text: t[0],
                        tokens: this.lexer.inline(t[0])
                    }
            }

            escape(e) {
                const t = this.rules.inline.escape.exec(e);
                if (t)
                    return {
                        type: "escape",
                        raw: t[0],
                        text: c(t[1])
                    }
            }

            tag(e) {
                const t = this.rules.inline.tag.exec(e);
                if (t)
                    return !this.lexer.state.inLink && /^<a /i.test(t[0]) ? this.lexer.state.inLink = !0 : this.lexer.state.inLink && /^<\/a>/i.test(t[0]) && (this.lexer.state.inLink = !1),
                        !this.lexer.state.inRawBlock && /^<(pre|code|kbd|script)(\s|>)/i.test(t[0]) ? this.lexer.state.inRawBlock = !0 : this.lexer.state.inRawBlock && /^<\/(pre|code|kbd|script)(\s|>)/i.test(t[0]) && (this.lexer.state.inRawBlock = !1),
                        {
                            type: "html",
                            raw: t[0],
                            inLink: this.lexer.state.inLink,
                            inRawBlock: this.lexer.state.inRawBlock,
                            block: !1,
                            text: t[0]
                        }
            }

            link(e) {
                const t = this.rules.inline.link.exec(e);
                if (t) {
                    const e = t[2].trim();
                    if (!this.options.pedantic && /^</.test(e)) {
                        if (!/>$/.test(e))
                            return;
                        const t = x(e.slice(0, -1), "\\");
                        if ((e.length - t.length) % 2 == 0)
                            return
                    } else {
                        const e = function (e, t) {
                            if (-1 === e.indexOf(t[1]))
                                return -1;
                            let n = 0;
                            for (let s = 0; s < e.length; s++)
                                if ("\\" === e[s])
                                    s++;
                                else if (e[s] === t[0])
                                    n++;
                                else if (e[s] === t[1] && (n--,
                                n < 0))
                                    return s;
                            return -1
                        }(t[2], "()");
                        if (e > -1) {
                            const n = (0 === t[0].indexOf("!") ? 5 : 4) + t[1].length + e;
                            t[2] = t[2].substring(0, e),
                                t[0] = t[0].substring(0, n).trim(),
                                t[3] = ""
                        }
                    }
                    let n = t[2]
                        , s = "";
                    if (this.options.pedantic) {
                        const e = /^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(n);
                        e && (n = e[1],
                            s = e[3])
                    } else
                        s = t[3] ? t[3].slice(1, -1) : "";
                    return n = n.trim(),
                    /^</.test(n) && (n = this.options.pedantic && !/>$/.test(e) ? n.slice(1) : n.slice(1, -1)),
                        b(t, {
                            href: n ? n.replace(this.rules.inline.anyPunctuation, "$1") : n,
                            title: s ? s.replace(this.rules.inline.anyPunctuation, "$1") : s
                        }, t[0], this.lexer)
                }
            }

            reflink(e, t) {
                let n;
                if ((n = this.rules.inline.reflink.exec(e)) || (n = this.rules.inline.nolink.exec(e))) {
                    const e = t[(n[2] || n[1]).replace(/\s+/g, " ").toLowerCase()];
                    if (!e) {
                        const e = n[0].charAt(0);
                        return {
                            type: "text",
                            raw: e,
                            text: e
                        }
                    }
                    return b(n, e, n[0], this.lexer)
                }
            }

            emStrong(e, t, n = "") {
                let s = this.rules.inline.emStrongLDelim.exec(e);
                if (!s)
                    return;
                if (s[3] && n.match(/[\p{L}\p{N}]/u))
                    return;
                if (!(s[1] || s[2] || "") || !n || this.rules.inline.punctuation.exec(n)) {
                    const n = [...s[0]].length - 1;
                    let r, i, l = n, o = 0;
                    const a = "*" === s[0][0] ? this.rules.inline.emStrongRDelimAst : this.rules.inline.emStrongRDelimUnd;
                    for (a.lastIndex = 0,
                             t = t.slice(-1 * e.length + n); null != (s = a.exec(t));) {
                        if (r = s[1] || s[2] || s[3] || s[4] || s[5] || s[6],
                            !r)
                            continue;
                        if (i = [...r].length,
                        s[3] || s[4]) {
                            l += i;
                            continue
                        }
                        if ((s[5] || s[6]) && n % 3 && !((n + i) % 3)) {
                            o += i;
                            continue
                        }
                        if (l -= i,
                        l > 0)
                            continue;
                        i = Math.min(i, i + l + o);
                        const t = [...s[0]][0].length
                            , a = e.slice(0, n + s.index + t + i);
                        if (Math.min(n, i) % 2) {
                            const e = a.slice(1, -1);
                            return {
                                type: "em",
                                raw: a,
                                text: e,
                                tokens: this.lexer.inlineTokens(e)
                            }
                        }
                        const c = a.slice(2, -2);
                        return {
                            type: "strong",
                            raw: a,
                            text: c,
                            tokens: this.lexer.inlineTokens(c)
                        }
                    }
                }
            }

            codespan(e) {
                const t = this.rules.inline.code.exec(e);
                if (t) {
                    let e = t[2].replace(/\n/g, " ");
                    const n = /[^ ]/.test(e)
                        , s = /^ /.test(e) && / $/.test(e);
                    return n && s && (e = e.substring(1, e.length - 1)),
                        e = c(e, !0),
                        {
                            type: "codespan",
                            raw: t[0],
                            text: e
                        }
                }
            }

            br(e) {
                const t = this.rules.inline.br.exec(e);
                if (t)
                    return {
                        type: "br",
                        raw: t[0]
                    }
            }

            del(e) {
                const t = this.rules.inline.del.exec(e);
                if (t)
                    return {
                        type: "del",
                        raw: t[0],
                        text: t[2],
                        tokens: this.lexer.inlineTokens(t[2])
                    }
            }

            autolink(e) {
                const t = this.rules.inline.autolink.exec(e);
                if (t) {
                    let e, n;
                    return "@" === t[2] ? (e = c(t[1]),
                        n = "mailto:" + e) : (e = c(t[1]),
                        n = e),
                        {
                            type: "link",
                            raw: t[0],
                            text: e,
                            href: n,
                            tokens: [{
                                type: "text",
                                raw: e,
                                text: e
                            }]
                        }
                }
            }

            url(e) {
                let t;
                if (t = this.rules.inline.url.exec(e)) {
                    let e, n;
                    if ("@" === t[2])
                        e = c(t[0]),
                            n = "mailto:" + e;
                    else {
                        let s;
                        do {
                            s = t[0],
                                t[0] = this.rules.inline._backpedal.exec(t[0])?.[0] ?? ""
                        } while (s !== t[0]);
                        e = c(t[0]),
                            n = "www." === t[1] ? "http://" + t[0] : t[0]
                    }
                    return {
                        type: "link",
                        raw: t[0],
                        text: e,
                        href: n,
                        tokens: [{
                            type: "text",
                            raw: e,
                            text: e
                        }]
                    }
                }
            }

            inlineText(e) {
                const t = this.rules.inline.text.exec(e);
                if (t) {
                    let e;
                    return e = this.lexer.state.inRawBlock ? t[0] : c(t[0]),
                        {
                            type: "text",
                            raw: t[0],
                            text: e
                        }
                }
            }
        }

        const m = /^ {0,3}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/
            , y = /(?:[*+-]|\d{1,9}[.)])/
            ,
            $ = k(/^(?!bull |blockCode|fences|blockquote|heading|html)((?:.|\n(?!\s*?\n|bull |blockCode|fences|blockquote|heading|html))+?)\n {0,3}(=+|-+) *(?:\n+|$)/).replace(/bull/g, y).replace(/blockCode/g, / {4}/).replace(/fences/g, / {0,3}(?:`{3,}|~{3,})/).replace(/blockquote/g, / {0,3}>/).replace(/heading/g, / {0,3}#{1,6}/).replace(/html/g, / {0,3}<[^\n>]+>\n/).getRegex()
            , z = /^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html|table| +\n)[^\n]+)*)/
            , T = /(?!\s*\])(?:\\.|[^\[\]\\])+/
            ,
            R = k(/^ {0,3}\[(label)\]: *(?:\n *)?([^<\s][^\s]*|<.*?>)(?:(?: +(?:\n *)?| *\n *)(title))? *(?:\n+|$)/).replace("label", T).replace("title", /(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/).getRegex()
            , _ = k(/^( {0,3}bull)([ \t][^\n]+?)?(?:\n|$)/).replace(/bull/g, y).getRegex()
            ,
            A = "address|article|aside|base|basefont|blockquote|body|caption|center|col|colgroup|dd|details|dialog|dir|div|dl|dt|fieldset|figcaption|figure|footer|form|frame|frameset|h[1-6]|head|header|hr|html|iframe|legend|li|link|main|menu|menuitem|meta|nav|noframes|ol|optgroup|option|p|param|search|section|summary|table|tbody|td|tfoot|th|thead|title|tr|track|ul"
            , S = /<!--(?:-?>|[\s\S]*?(?:-->|$))/
            ,
            I = k("^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|<![A-Z][\\s\\S]*?(?:>\\n*|$)|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:(?:\\n *)+\\n|$)|<(?!script|pre|style|textarea)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$)|</(?!script|pre|style|textarea)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$))", "i").replace("comment", S).replace("tag", A).replace("attribute", / +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex()
            ,
            E = k(z).replace("hr", m).replace("heading", " {0,3}#{1,6}(?:\\s|$)").replace("|lheading", "").replace("|table", "").replace("blockquote", " {0,3}>").replace("fences", " {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list", " {0,3}(?:[*+-]|1[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", A).getRegex()
            , q = {
                blockquote: k(/^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/).replace("paragraph", E).getRegex(),
                code: /^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/,
                def: R,
                fences: /^ {0,3}(`{3,}(?=[^`\n]*(?:\n|$))|~{3,})([^\n]*)(?:\n|$)(?:|([\s\S]*?)(?:\n|$))(?: {0,3}\1[~`]* *(?=\n|$)|$)/,
                heading: /^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,
                hr: m,
                html: I,
                lheading: $,
                list: _,
                newline: /^(?: *(?:\n|$))+/,
                paragraph: E,
                table: f,
                text: /^[^\n]+/
            }
            ,
            Z = k("^ *([^\\n ].*)\\n {0,3}((?:\\| *)?:?-+:? *(?:\\| *:?-+:? *)*(?:\\| *)?)(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)").replace("hr", m).replace("heading", " {0,3}#{1,6}(?:\\s|$)").replace("blockquote", " {0,3}>").replace("code", " {4}[^\\n]").replace("fences", " {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list", " {0,3}(?:[*+-]|1[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", A).getRegex()
            , L = {
                ...q,
                table: Z,
                paragraph: k(z).replace("hr", m).replace("heading", " {0,3}#{1,6}(?:\\s|$)").replace("|lheading", "").replace("table", Z).replace("blockquote", " {0,3}>").replace("fences", " {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list", " {0,3}(?:[*+-]|1[.)]) ").replace("html", "</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag", A).getRegex()
            }
            , P = {
                ...q,
                html: k("^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)|<tag(?:\"[^\"]*\"|'[^']*'|\\s[^'\"/>\\s]*)*?/?> *(?:\\n{2,}|\\s*$))").replace("comment", S).replace(/tag/g, "(?!(?:a|em|strong|small|s|cite|q|dfn|abbr|data|time|code|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo|span|br|wbr|ins|del|img)\\b)\\w+(?!:|[^\\w\\s@]*@)\\b").getRegex(),
                def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,
                heading: /^(#{1,6})(.*)(?:\n+|$)/,
                fences: f,
                lheading: /^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,
                paragraph: k(z).replace("hr", m).replace("heading", " *#{1,6} *[^\n]").replace("lheading", $).replace("|table", "").replace("blockquote", " {0,3}>").replace("|fences", "").replace("|list", "").replace("|html", "").replace("|tag", "").getRegex()
            }
            , Q = /^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/
            , v = /^( {2,}|\\)\n(?!\s*$)/
            , B = "\\p{P}\\p{S}"
            , C = k(/^((?![*_])[\spunctuation])/, "u").replace(/punctuation/g, B).getRegex()
            ,
            M = k(/^(?:\*+(?:((?!\*)[punct])|[^\s*]))|^_+(?:((?!_)[punct])|([^\s_]))/, "u").replace(/punct/g, B).getRegex()
            ,
            O = k("^[^_*]*?__[^_*]*?\\*[^_*]*?(?=__)|[^*]+(?=[^*])|(?!\\*)[punct](\\*+)(?=[\\s]|$)|[^punct\\s](\\*+)(?!\\*)(?=[punct\\s]|$)|(?!\\*)[punct\\s](\\*+)(?=[^punct\\s])|[\\s](\\*+)(?!\\*)(?=[punct])|(?!\\*)[punct](\\*+)(?!\\*)(?=[punct])|[^punct\\s](\\*+)(?=[^punct\\s])", "gu").replace(/punct/g, B).getRegex()
            ,
            D = k("^[^_*]*?\\*\\*[^_*]*?_[^_*]*?(?=\\*\\*)|[^_]+(?=[^_])|(?!_)[punct](_+)(?=[\\s]|$)|[^punct\\s](_+)(?!_)(?=[punct\\s]|$)|(?!_)[punct\\s](_+)(?=[^punct\\s])|[\\s](_+)(?!_)(?=[punct])|(?!_)[punct](_+)(?!_)(?=[punct])", "gu").replace(/punct/g, B).getRegex()
            , j = k(/\\([punct])/, "gu").replace(/punct/g, B).getRegex()
            ,
            H = k(/^<(scheme:[^\s\x00-\x1f<>]*|email)>/).replace("scheme", /[a-zA-Z][a-zA-Z0-9+.-]{1,31}/).replace("email", /[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+(@)[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+(?![-_])/).getRegex()
            , U = k(S).replace("(?:--\x3e|$)", "--\x3e").getRegex()
            ,
            X = k("^comment|^</[a-zA-Z][\\w:-]*\\s*>|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^<![a-zA-Z]+\\s[\\s\\S]*?>|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>").replace("comment", U).replace("attribute", /\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/).getRegex()
            , F = /(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/
            ,
            N = k(/^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/).replace("label", F).replace("href", /<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/).replace("title", /"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/).getRegex()
            , G = k(/^!?\[(label)\]\[(ref)\]/).replace("label", F).replace("ref", T).getRegex()
            , J = k(/^!?\[(ref)\](?:\[\])?/).replace("ref", T).getRegex()
            , K = {
                _backpedal: f,
                anyPunctuation: j,
                autolink: H,
                blockSkip: /\[[^[\]]*?\]\([^\(\)]*?\)|`[^`]*?`|<[^<>]*?>/g,
                br: v,
                code: /^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,
                del: f,
                emStrongLDelim: M,
                emStrongRDelimAst: O,
                emStrongRDelimUnd: D,
                escape: Q,
                link: N,
                nolink: J,
                punctuation: C,
                reflink: G,
                reflinkSearch: k("reflink|nolink(?!\\()", "g").replace("reflink", G).replace("nolink", J).getRegex(),
                tag: X,
                text: /^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*_]|\b_|$)|[^ ](?= {2,}\n)))/,
                url: f
            }
            , V = {
                ...K,
                link: k(/^!?\[(label)\]\((.*?)\)/).replace("label", F).getRegex(),
                reflink: k(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label", F).getRegex()
            }
            , W = {
                ...K,
                escape: k(Q).replace("])", "~|])").getRegex(),
                url: k(/^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/, "i").replace("email", /[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/).getRegex(),
                _backpedal: /(?:[^?!.,:;*_'"~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_'"~)]+(?!$))+/,
                del: /^(~~?)(?=[^\s~])([\s\S]*?[^\s~])\1(?=[^~]|$)/,
                text: /^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\<!\[`*~_]|\b_|https?:\/\/|ftp:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)))/
            }
            , Y = {
                ...W,
                br: k(v).replace("{2,}", "*").getRegex(),
                text: k(W.text).replace("\\b_", "\\b_| {2,}\\n").replace(/\{2,\}/g, "*").getRegex()
            }
            , ee = {
                normal: q,
                gfm: L,
                pedantic: P
            }
            , te = {
                normal: K,
                gfm: W,
                breaks: Y,
                pedantic: V
            };

        class ne {
            tokens;
            options;
            state;
            tokenizer;
            inlineQueue;

            constructor(t) {
                this.tokens = [],
                    this.tokens.links = Object.create(null),
                    this.options = t || e.defaults,
                    this.options.tokenizer = this.options.tokenizer || new w,
                    this.tokenizer = this.options.tokenizer,
                    this.tokenizer.options = this.options,
                    this.tokenizer.lexer = this,
                    this.inlineQueue = [],
                    this.state = {
                        inLink: !1,
                        inRawBlock: !1,
                        top: !0
                    };
                const n = {
                    block: ee.normal,
                    inline: te.normal
                };
                this.options.pedantic ? (n.block = ee.pedantic,
                    n.inline = te.pedantic) : this.options.gfm && (n.block = ee.gfm,
                    this.options.breaks ? n.inline = te.breaks : n.inline = te.gfm),
                    this.tokenizer.rules = n
            }

            static get rules() {
                return {
                    block: ee,
                    inline: te
                }
            }

            static lex(e, t) {
                return new ne(t).lex(e)
            }

            static lexInline(e, t) {
                return new ne(t).inlineTokens(e)
            }

            lex(e) {
                e = e.replace(/\r\n|\r/g, "\n"),
                    this.blockTokens(e, this.tokens);
                for (let e = 0; e < this.inlineQueue.length; e++) {
                    const t = this.inlineQueue[e];
                    this.inlineTokens(t.src, t.tokens)
                }
                return this.inlineQueue = [],
                    this.tokens
            }

            blockTokens(e, t = []) {
                let n, s, r, i;
                for (e = this.options.pedantic ? e.replace(/\t/g, "    ").replace(/^ +$/gm, "") : e.replace(/^( *)(\t+)/gm, ((e, t, n) => t + "    ".repeat(n.length))); e;)
                    if (!(this.options.extensions && this.options.extensions.block && this.options.extensions.block.some((s => !!(n = s.call({
                        lexer: this
                    }, e, t)) && (e = e.substring(n.raw.length),
                        t.push(n),
                        !0)))))
                        if (n = this.tokenizer.space(e))
                            e = e.substring(n.raw.length),
                                1 === n.raw.length && t.length > 0 ? t[t.length - 1].raw += "\n" : t.push(n);
                        else if (n = this.tokenizer.code(e))
                            e = e.substring(n.raw.length),
                                s = t[t.length - 1],
                                !s || "paragraph" !== s.type && "text" !== s.type ? t.push(n) : (s.raw += "\n" + n.raw,
                                    s.text += "\n" + n.text,
                                    this.inlineQueue[this.inlineQueue.length - 1].src = s.text);
                        else if (n = this.tokenizer.fences(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.heading(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.hr(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.blockquote(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.list(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.html(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.def(e))
                            e = e.substring(n.raw.length),
                                s = t[t.length - 1],
                                !s || "paragraph" !== s.type && "text" !== s.type ? this.tokens.links[n.tag] || (this.tokens.links[n.tag] = {
                                    href: n.href,
                                    title: n.title
                                }) : (s.raw += "\n" + n.raw,
                                    s.text += "\n" + n.raw,
                                    this.inlineQueue[this.inlineQueue.length - 1].src = s.text);
                        else if (n = this.tokenizer.table(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.lheading(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else {
                            if (r = e,
                            this.options.extensions && this.options.extensions.startBlock) {
                                let t = 1 / 0;
                                const n = e.slice(1);
                                let s;
                                this.options.extensions.startBlock.forEach((e => {
                                        s = e.call({
                                            lexer: this
                                        }, n),
                                        "number" == typeof s && s >= 0 && (t = Math.min(t, s))
                                    }
                                )),
                                t < 1 / 0 && t >= 0 && (r = e.substring(0, t + 1))
                            }
                            if (this.state.top && (n = this.tokenizer.paragraph(r)))
                                s = t[t.length - 1],
                                    i && "paragraph" === s.type ? (s.raw += "\n" + n.raw,
                                        s.text += "\n" + n.text,
                                        this.inlineQueue.pop(),
                                        this.inlineQueue[this.inlineQueue.length - 1].src = s.text) : t.push(n),
                                    i = r.length !== e.length,
                                    e = e.substring(n.raw.length);
                            else if (n = this.tokenizer.text(e))
                                e = e.substring(n.raw.length),
                                    s = t[t.length - 1],
                                    s && "text" === s.type ? (s.raw += "\n" + n.raw,
                                        s.text += "\n" + n.text,
                                        this.inlineQueue.pop(),
                                        this.inlineQueue[this.inlineQueue.length - 1].src = s.text) : t.push(n);
                            else if (e) {
                                const t = "Infinite loop on byte: " + e.charCodeAt(0);
                                if (this.options.silent) {
                                    console.error(t);
                                    break
                                }
                                throw new Error(t)
                            }
                        }
                return this.state.top = !0,
                    t
            }

            inline(e, t = []) {
                return this.inlineQueue.push({
                    src: e,
                    tokens: t
                }),
                    t
            }

            inlineTokens(e, t = []) {
                let n, s, r, i, l, o, a = e;
                if (this.tokens.links) {
                    const e = Object.keys(this.tokens.links);
                    if (e.length > 0)
                        for (; null != (i = this.tokenizer.rules.inline.reflinkSearch.exec(a));)
                            e.includes(i[0].slice(i[0].lastIndexOf("[") + 1, -1)) && (a = a.slice(0, i.index) + "[" + "a".repeat(i[0].length - 2) + "]" + a.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex))
                }
                for (; null != (i = this.tokenizer.rules.inline.blockSkip.exec(a));)
                    a = a.slice(0, i.index) + "[" + "a".repeat(i[0].length - 2) + "]" + a.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);
                for (; null != (i = this.tokenizer.rules.inline.anyPunctuation.exec(a));)
                    a = a.slice(0, i.index) + "++" + a.slice(this.tokenizer.rules.inline.anyPunctuation.lastIndex);
                for (; e;)
                    if (l || (o = ""),
                        l = !1,
                        !(this.options.extensions && this.options.extensions.inline && this.options.extensions.inline.some((s => !!(n = s.call({
                            lexer: this
                        }, e, t)) && (e = e.substring(n.raw.length),
                            t.push(n),
                            !0)))))
                        if (n = this.tokenizer.escape(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.tag(e))
                            e = e.substring(n.raw.length),
                                s = t[t.length - 1],
                                s && "text" === n.type && "text" === s.type ? (s.raw += n.raw,
                                    s.text += n.text) : t.push(n);
                        else if (n = this.tokenizer.link(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.reflink(e, this.tokens.links))
                            e = e.substring(n.raw.length),
                                s = t[t.length - 1],
                                s && "text" === n.type && "text" === s.type ? (s.raw += n.raw,
                                    s.text += n.text) : t.push(n);
                        else if (n = this.tokenizer.emStrong(e, a, o))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.codespan(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.br(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.del(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (n = this.tokenizer.autolink(e))
                            e = e.substring(n.raw.length),
                                t.push(n);
                        else if (this.state.inLink || !(n = this.tokenizer.url(e))) {
                            if (r = e,
                            this.options.extensions && this.options.extensions.startInline) {
                                let t = 1 / 0;
                                const n = e.slice(1);
                                let s;
                                this.options.extensions.startInline.forEach((e => {
                                        s = e.call({
                                            lexer: this
                                        }, n),
                                        "number" == typeof s && s >= 0 && (t = Math.min(t, s))
                                    }
                                )),
                                t < 1 / 0 && t >= 0 && (r = e.substring(0, t + 1))
                            }
                            if (n = this.tokenizer.inlineText(r))
                                e = e.substring(n.raw.length),
                                "_" !== n.raw.slice(-1) && (o = n.raw.slice(-1)),
                                    l = !0,
                                    s = t[t.length - 1],
                                    s && "text" === s.type ? (s.raw += n.raw,
                                        s.text += n.text) : t.push(n);
                            else if (e) {
                                const t = "Infinite loop on byte: " + e.charCodeAt(0);
                                if (this.options.silent) {
                                    console.error(t);
                                    break
                                }
                                throw new Error(t)
                            }
                        } else
                            e = e.substring(n.raw.length),
                                t.push(n);
                return t
            }
        }

        class se {
            options;

            constructor(t) {
                this.options = t || e.defaults
            }

            code(e, t, n) {
                const s = (t || "").match(/^\S*/)?.[0];
                return e = e.replace(/\n$/, "") + "\n",
                    s ? '<pre><code class="language-' + c(s) + '">' + (n ? e : c(e, !0)) + "</code></pre>\n" : "<pre><code>" + (n ? e : c(e, !0)) + "</code></pre>\n"
            }

            blockquote(e) {
                return `<blockquote>\n${e}</blockquote>\n`
            }

            html(e, t) {
                return e
            }

            heading(e, t, n) {
                return `<h${t}>${e}</h${t}>\n`
            }

            hr() {
                return "<hr>\n"
            }

            list(e, t, n) {
                const s = t ? "ol" : "ul";
                return "<" + s + (t && 1 !== n ? ' start="' + n + '"' : "") + ">\n" + e + "</" + s + ">\n"
            }

            listitem(e, t, n) {
                return `<li>${e}</li>\n`
            }

            checkbox(e) {
                return "<input " + (e ? 'checked="" ' : "") + 'disabled="" type="checkbox">'
            }

            paragraph(e) {
                return `<p>${e}</p>\n`
            }

            table(e, t) {
                return t && (t = `<tbody>${t}</tbody>`),
                "<table>\n<thead>\n" + e + "</thead>\n" + t + "</table>\n"
            }

            tablerow(e) {
                return `<tr>\n${e}</tr>\n`
            }

            tablecell(e, t) {
                const n = t.header ? "th" : "td";
                return (t.align ? `<${n} align="${t.align}">` : `<${n}>`) + e + `</${n}>\n`
            }

            strong(e) {
                return `<strong>${e}</strong>`
            }

            em(e) {
                return `<em>${e}</em>`
            }

            codespan(e) {
                return `<code>${e}</code>`
            }

            br() {
                return "<br>"
            }

            del(e) {
                return `<del>${e}</del>`
            }

            link(e, t, n) {
                const s = g(e);
                if (null === s)
                    return n;
                let r = '<a href="' + (e = s) + '"';
                return t && (r += ' title="' + t + '"'),
                    r += ">" + n + "</a>",
                    r
            }

            image(e, t, n) {
                const s = g(e);
                if (null === s)
                    return n;
                let r = `<img src="${e = s}" alt="${n}"`;
                return t && (r += ` title="${t}"`),
                    r += ">",
                    r
            }

            text(e) {
                return e
            }
        }

        class re {
            strong(e) {
                return e
            }

            em(e) {
                return e
            }

            codespan(e) {
                return e
            }

            del(e) {
                return e
            }

            html(e) {
                return e
            }

            text(e) {
                return e
            }

            link(e, t, n) {
                return "" + n
            }

            image(e, t, n) {
                return "" + n
            }

            br() {
                return ""
            }
        }

        class ie {
            options;
            renderer;
            textRenderer;

            constructor(t) {
                this.options = t || e.defaults,
                    this.options.renderer = this.options.renderer || new se,
                    this.renderer = this.options.renderer,
                    this.renderer.options = this.options,
                    this.textRenderer = new re
            }

            static parse(e, t) {
                return new ie(t).parse(e)
            }

            static parseInline(e, t) {
                return new ie(t).parseInline(e)
            }

            parse(e, t = !0) {
                let n = "";
                for (let s = 0; s < e.length; s++) {
                    const r = e[s];
                    if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[r.type]) {
                        const e = r
                            , t = this.options.extensions.renderers[e.type].call({
                            parser: this
                        }, e);
                        if (!1 !== t || !["space", "hr", "heading", "code", "table", "blockquote", "list", "html", "paragraph", "text"].includes(e.type)) {
                            n += t || "";
                            continue
                        }
                    }
                    switch (r.type) {
                        case "space":
                            continue;
                        case "hr":
                            n += this.renderer.hr();
                            continue;
                        case "heading": {
                            const e = r;
                            n += this.renderer.heading(this.parseInline(e.tokens), e.depth, p(this.parseInline(e.tokens, this.textRenderer)));
                            continue
                        }
                        case "code": {
                            const e = r;
                            n += this.renderer.code(e.text, e.lang, !!e.escaped);
                            continue
                        }
                        case "table": {
                            const e = r;
                            let t = ""
                                , s = "";
                            for (let t = 0; t < e.header.length; t++)
                                s += this.renderer.tablecell(this.parseInline(e.header[t].tokens), {
                                    header: !0,
                                    align: e.align[t]
                                });
                            t += this.renderer.tablerow(s);
                            let i = "";
                            for (let t = 0; t < e.rows.length; t++) {
                                const n = e.rows[t];
                                s = "";
                                for (let t = 0; t < n.length; t++)
                                    s += this.renderer.tablecell(this.parseInline(n[t].tokens), {
                                        header: !1,
                                        align: e.align[t]
                                    });
                                i += this.renderer.tablerow(s)
                            }
                            n += this.renderer.table(t, i);
                            continue
                        }
                        case "blockquote": {
                            const e = r
                                , t = this.parse(e.tokens);
                            n += this.renderer.blockquote(t);
                            continue
                        }
                        case "list": {
                            const e = r
                                , t = e.ordered
                                , s = e.start
                                , i = e.loose;
                            let l = "";
                            for (let t = 0; t < e.items.length; t++) {
                                const n = e.items[t]
                                    , s = n.checked
                                    , r = n.task;
                                let o = "";
                                if (n.task) {
                                    const e = this.renderer.checkbox(!!s);
                                    i ? n.tokens.length > 0 && "paragraph" === n.tokens[0].type ? (n.tokens[0].text = e + " " + n.tokens[0].text,
                                    n.tokens[0].tokens && n.tokens[0].tokens.length > 0 && "text" === n.tokens[0].tokens[0].type && (n.tokens[0].tokens[0].text = e + " " + n.tokens[0].tokens[0].text)) : n.tokens.unshift({
                                        type: "text",
                                        text: e + " "
                                    }) : o += e + " "
                                }
                                o += this.parse(n.tokens, i),
                                    l += this.renderer.listitem(o, r, !!s)
                            }
                            n += this.renderer.list(l, t, s);
                            continue
                        }
                        case "html": {
                            const e = r;
                            n += this.renderer.html(e.text, e.block);
                            continue
                        }
                        case "paragraph": {
                            const e = r;
                            n += this.renderer.paragraph(this.parseInline(e.tokens));
                            continue
                        }
                        case "text": {
                            let i = r
                                , l = i.tokens ? this.parseInline(i.tokens) : i.text;
                            for (; s + 1 < e.length && "text" === e[s + 1].type;)
                                i = e[++s],
                                    l += "\n" + (i.tokens ? this.parseInline(i.tokens) : i.text);
                            n += t ? this.renderer.paragraph(l) : l;
                            continue
                        }
                        default: {
                            const e = 'Token with "' + r.type + '" type was not found.';
                            if (this.options.silent)
                                return console.error(e),
                                    "";
                            throw new Error(e)
                        }
                    }
                }
                return n
            }

            parseInline(e, t) {
                t = t || this.renderer;
                let n = "";
                for (let s = 0; s < e.length; s++) {
                    const r = e[s];
                    if (this.options.extensions && this.options.extensions.renderers && this.options.extensions.renderers[r.type]) {
                        const e = this.options.extensions.renderers[r.type].call({
                            parser: this
                        }, r);
                        if (!1 !== e || !["escape", "html", "link", "image", "strong", "em", "codespan", "br", "del", "text"].includes(r.type)) {
                            n += e || "";
                            continue
                        }
                    }
                    switch (r.type) {
                        case "escape": {
                            const e = r;
                            n += t.text(e.text);
                            break
                        }
                        case "html": {
                            const e = r;
                            n += t.html(e.text);
                            break
                        }
                        case "link": {
                            const e = r;
                            n += t.link(e.href, e.title, this.parseInline(e.tokens, t));
                            break
                        }
                        case "image": {
                            const e = r;
                            n += t.image(e.href, e.title, e.text);
                            break
                        }
                        case "strong": {
                            const e = r;
                            n += t.strong(this.parseInline(e.tokens, t));
                            break
                        }
                        case "em": {
                            const e = r;
                            n += t.em(this.parseInline(e.tokens, t));
                            break
                        }
                        case "codespan": {
                            const e = r;
                            n += t.codespan(e.text);
                            break
                        }
                        case "br":
                            n += t.br();
                            break;
                        case "del": {
                            const e = r;
                            n += t.del(this.parseInline(e.tokens, t));
                            break
                        }
                        case "text": {
                            const e = r;
                            n += t.text(e.text);
                            break
                        }
                        default: {
                            const e = 'Token with "' + r.type + '" type was not found.';
                            if (this.options.silent)
                                return console.error(e),
                                    "";
                            throw new Error(e)
                        }
                    }
                }
                return n
            }
        }

        class le {
            options;

            constructor(t) {
                this.options = t || e.defaults
            }

            static passThroughHooks = new Set(["preprocess", "postprocess", "processAllTokens"]);

            preprocess(e) {
                return e
            }

            postprocess(e) {
                return e
            }

            processAllTokens(e) {
                return e
            }
        }

        class oe {
            defaults = {
                async: !1,
                breaks: !1,
                extensions: null,
                gfm: !0,
                hooks: null,
                pedantic: !1,
                renderer: null,
                silent: !1,
                tokenizer: null,
                walkTokens: null
            };
            options = this.setOptions;
            parse = this.#e(ne.lex, ie.parse);
            parseInline = this.#e(ne.lexInline, ie.parseInline);
            Parser = ie;
            Renderer = se;
            TextRenderer = re;
            Lexer = ne;
            Tokenizer = w;
            Hooks = le;

            constructor(...e) {
                this.use(...e)
            }

            walkTokens(e, t) {
                let n = [];
                for (const s of e)
                    switch (n = n.concat(t.call(this, s)),
                        s.type) {
                        case "table": {
                            const e = s;
                            for (const s of e.header)
                                n = n.concat(this.walkTokens(s.tokens, t));
                            for (const s of e.rows)
                                for (const e of s)
                                    n = n.concat(this.walkTokens(e.tokens, t));
                            break
                        }
                        case "list": {
                            const e = s;
                            n = n.concat(this.walkTokens(e.items, t));
                            break
                        }
                        default: {
                            const e = s;
                            this.defaults.extensions?.childTokens?.[e.type] ? this.defaults.extensions.childTokens[e.type].forEach((s => {
                                    const r = e[s].flat(1 / 0);
                                    n = n.concat(this.walkTokens(r, t))
                                }
                            )) : e.tokens && (n = n.concat(this.walkTokens(e.tokens, t)))
                        }
                    }
                return n
            }

            use(...e) {
                const t = this.defaults.extensions || {
                    renderers: {},
                    childTokens: {}
                };
                return e.forEach((e => {
                        const n = {
                            ...e
                        };
                        if (n.async = this.defaults.async || n.async || !1,
                        e.extensions && (e.extensions.forEach((e => {
                                if (!e.name)
                                    throw new Error("extension name required");
                                if ("renderer" in e) {
                                    const n = t.renderers[e.name];
                                    t.renderers[e.name] = n ? function (...t) {
                                            let s = e.renderer.apply(this, t);
                                            return !1 === s && (s = n.apply(this, t)),
                                                s
                                        }
                                        : e.renderer
                                }
                                if ("tokenizer" in e) {
                                    if (!e.level || "block" !== e.level && "inline" !== e.level)
                                        throw new Error("extension level must be 'block' or 'inline'");
                                    const n = t[e.level];
                                    n ? n.unshift(e.tokenizer) : t[e.level] = [e.tokenizer],
                                    e.start && ("block" === e.level ? t.startBlock ? t.startBlock.push(e.start) : t.startBlock = [e.start] : "inline" === e.level && (t.startInline ? t.startInline.push(e.start) : t.startInline = [e.start]))
                                }
                                "childTokens" in e && e.childTokens && (t.childTokens[e.name] = e.childTokens)
                            }
                        )),
                            n.extensions = t),
                            e.renderer) {
                            const t = this.defaults.renderer || new se(this.defaults);
                            for (const n in e.renderer) {
                                if (!(n in t))
                                    throw new Error(`renderer '${n}' does not exist`);
                                if ("options" === n)
                                    continue;
                                const s = n
                                    , r = e.renderer[s]
                                    , i = t[s];
                                t[s] = (...e) => {
                                    let n = r.apply(t, e);
                                    return !1 === n && (n = i.apply(t, e)),
                                    n || ""
                                }
                            }
                            n.renderer = t
                        }
                        if (e.tokenizer) {
                            const t = this.defaults.tokenizer || new w(this.defaults);
                            for (const n in e.tokenizer) {
                                if (!(n in t))
                                    throw new Error(`tokenizer '${n}' does not exist`);
                                if (["options", "rules", "lexer"].includes(n))
                                    continue;
                                const s = n
                                    , r = e.tokenizer[s]
                                    , i = t[s];
                                t[s] = (...e) => {
                                    let n = r.apply(t, e);
                                    return !1 === n && (n = i.apply(t, e)),
                                        n
                                }
                            }
                            n.tokenizer = t
                        }
                        if (e.hooks) {
                            const t = this.defaults.hooks || new le;
                            for (const n in e.hooks) {
                                if (!(n in t))
                                    throw new Error(`hook '${n}' does not exist`);
                                if ("options" === n)
                                    continue;
                                const s = n
                                    , r = e.hooks[s]
                                    , i = t[s];
                                le.passThroughHooks.has(n) ? t[s] = e => {
                                        if (this.defaults.async)
                                            return Promise.resolve(r.call(t, e)).then((e => i.call(t, e)));
                                        const n = r.call(t, e);
                                        return i.call(t, n)
                                    }
                                    : t[s] = (...e) => {
                                        let n = r.apply(t, e);
                                        return !1 === n && (n = i.apply(t, e)),
                                            n
                                    }
                            }
                            n.hooks = t
                        }
                        if (e.walkTokens) {
                            const t = this.defaults.walkTokens
                                , s = e.walkTokens;
                            n.walkTokens = function (e) {
                                let n = [];
                                return n.push(s.call(this, e)),
                                t && (n = n.concat(t.call(this, e))),
                                    n
                            }
                        }
                        this.defaults = {
                            ...this.defaults,
                            ...n
                        }
                    }
                )),
                    this
            }

            setOptions(e) {
                return this.defaults = {
                    ...this.defaults,
                    ...e
                },
                    this
            }

            lexer(e, t) {
                return ne.lex(e, t ?? this.defaults)
            }

            parser(e, t) {
                return ie.parse(e, t ?? this.defaults)
            }

            #e(e, t) {
                return (n, s) => {
                    const r = {
                        ...s
                    }
                        , i = {
                        ...this.defaults,
                        ...r
                    };
                    !0 === this.defaults.async && !1 === r.async && (i.silent || console.warn("marked(): The async option was set to true by an extension. The async: false option sent to parse will be ignored."),
                        i.async = !0);
                    const l = this.#t(!!i.silent, !!i.async);
                    if (null == n)
                        return l(new Error("marked(): input parameter is undefined or null"));
                    if ("string" != typeof n)
                        return l(new Error("marked(): input parameter is of type " + Object.prototype.toString.call(n) + ", string expected"));
                    if (i.hooks && (i.hooks.options = i),
                        i.async)
                        return Promise.resolve(i.hooks ? i.hooks.preprocess(n) : n).then((t => e(t, i))).then((e => i.hooks ? i.hooks.processAllTokens(e) : e)).then((e => i.walkTokens ? Promise.all(this.walkTokens(e, i.walkTokens)).then((() => e)) : e)).then((e => t(e, i))).then((e => i.hooks ? i.hooks.postprocess(e) : e)).catch(l);
                    try {
                        i.hooks && (n = i.hooks.preprocess(n));
                        let s = e(n, i);
                        i.hooks && (s = i.hooks.processAllTokens(s)),
                        i.walkTokens && this.walkTokens(s, i.walkTokens);
                        let r = t(s, i);
                        return i.hooks && (r = i.hooks.postprocess(r)),
                            r
                    } catch (e) {
                        return l(e)
                    }
                }
            }

            #t(e, t) {
                return n => {
                    if (n.message += "\nPlease report this to https://github.com/markedjs/marked.",
                        e) {
                        const e = "<p>An error occurred:</p><pre>" + c(n.message + "", !0) + "</pre>";
                        return t ? Promise.resolve(e) : e
                    }
                    if (t)
                        return Promise.reject(n);
                    throw n
                }
            }
        }

        const ae = new oe;

        function ce(e, t) {
            return ae.parse(e, t)
        }

        ce.options = ce.setOptions = function (e) {
            return ae.setOptions(e),
                ce.defaults = ae.defaults,
                n(ce.defaults),
                ce
        }
            ,
            ce.getDefaults = t,
            ce.defaults = e.defaults,
            ce.use = function (...e) {
                return ae.use(...e),
                    ce.defaults = ae.defaults,
                    n(ce.defaults),
                    ce
            }
            ,
            ce.walkTokens = function (e, t) {
                return ae.walkTokens(e, t)
            }
            ,
            ce.parseInline = ae.parseInline,
            ce.Parser = ie,
            ce.parser = ie.parse,
            ce.Renderer = se,
            ce.TextRenderer = re,
            ce.Lexer = ne,
            ce.lexer = ne.lex,
            ce.Tokenizer = w,
            ce.Hooks = le,
            ce.parse = ce;
        const he = ce.options
            , pe = ce.setOptions
            , ue = ce.use
            , ke = ce.walkTokens
            , ge = ce.parseInline
            , fe = ce
            , de = ie.parse
            , xe = ne.lex;
        e.Hooks = le,
            e.Lexer = ne,
            e.Marked = oe,
            e.Parser = ie,
            e.Renderer = se,
            e.TextRenderer = re,
            e.Tokenizer = w,
            e.getDefaults = t,
            e.lexer = xe,
            e.marked = ce,
            e.options = he,
            e.parse = fe,
            e.parseInline = ge,
            e.parser = de,
            e.setOptions = pe,
            e.use = ue,
            e.walkTokens = ke
    }
));

启动服务,浏览器打开 index.html
在这里插入图片描述

开通DashScope灵积模型服务

申请通义API-KEY:
https://help.aliyun.com/zh/dashscope/developer-reference/activate-dashscope-and-create-an-api-key
在这里插入图片描述

代码编译过程中会遇到如下图所示问题
在这里插入图片描述
解决办法
找到自己本地Maven的setting.xml,移除阿里云镜像仓库地址
在这里插入图片描述

pom引入官方仓库

<repositories>
  <repository>
    <id>spring-milestones</id>
    <name>Spring Milestones</name>
    <url>https://repo.spring.io/milestone</url>
    <snapshots>
        <enabled>false</enabled>
    </snapshots>
  </repository>
</repositories>

重新编辑,问题解决
公众号推荐

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1954057.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

理解 HTTP 请求中 Query 和 Body 的异同

本文将深入探讨HTTP请求中的两个关键要素&#xff1a;查询参数&#xff08;Query&#xff09;和请求体&#xff08;Body&#xff09;。我们将阐明它们之间的差异&#xff0c;并讨论在何种情况下使用每一种。 HTTP 请求概述 HTTP 请求是客户端&#xff08;如浏览器&#xff09…

知道秘密的人

一、力扣题目&#xff1a; 二、理论分析 由于 天数是一天一天变化的&#xff0c;用 数组的下标代表天数i, 数组中的 数据代表知道秘密在第i天的人数 假设在某个人在知道秘密的第3天开始传播&#xff0c;在第6天忘记&#xff0c;由于 第1天1个人发现了秘密 spread为能传播秘密的…

MATLAB-bode图编程

num[1 1];den [2 1];tf(num,den)bode(tf(num,den));hold on

【QT】TCP

目录 核心API 示例&#xff1a;服务器和客户端信息互发 服务器代码实现 第一步&#xff1a;创建QTcpServer对象的实例 第二步&#xff1a;绑定信号槽&#xff0c;处理新的连接 第三步&#xff1a;绑定并监听端口号 客户端代码实现 第一步&#xff1a;创建socket对象的实…

【计算机网络】WireShark和简单http抓包实验

一&#xff1a;实验目的 1&#xff1a;熟悉WireShark的安装流程和界面操作流程。 2&#xff1a;学会简单http的抓取和过滤&#xff0c;并分析导出结果。 二&#xff1a;实验仪器设备及软件 硬件&#xff1a; Windows 2019操作系统的计算机等。 软件&#xff1a;WireShark、…

智能PDF转markdown

嘿&#xff0c;各位技术大咖们&#xff0c;今天我要给大家带来一个超酷的项目——“智能PDF转Markdown”&#xff0c;这可是数字化办公的神器&#xff0c;基于Marker技术&#xff0c;让PDF文档秒变Markdown&#xff0c;轻松应对RAG知识库构建等任务的挑战&#xff01; 打造了一…

Python | Leetcode Python题解之第299题猜数字游戏

题目&#xff1a; 题解&#xff1a; class Solution:def getHint(self, secret: str, guess: str) -> str:bulls 0cntS, cntG [0] * 10, [0] * 10for s, g in zip(secret, guess):if s g:bulls 1else:cntS[int(s)] 1cntG[int(g)] 1cows sum(min(s, g) for s, g in z…

学习笔记 韩顺平 零基础30天学会Java(2024.7.22)

P407 接口使用细节2 P407 接口课堂练习 对于最后一个的输出&#xff1a;B因为实现了A的接口&#xff0c;所以和继承一样&#xff0c;B可以访问A的变量 P409 接口VS继承 接口对单继承机制&#xff08;是指只能继承一个类&#xff09;进行了补充 也可以理解为&#xff0c;子类通过…

IndexError: list index out of range

IndexError: list index out of range 欢迎来到我的主页&#xff0c;我是博主英杰&#xff0c;211科班出身&#xff0c;就职于医疗科技公司&#xff0c;热衷分享知识&#xff0c;武汉城市开发者社区主理人 擅长.net、C、python开发&#xff0c; 如果遇到技术问题&#xff0c;即…

锅总介绍CNCF主要目标、全景图及发展历史

一、CNCF简介 云原生计算基金会&#xff08;Cloud Native Computing Foundation&#xff0c;简称 CNCF&#xff09;是一个成立于 2015 年的非营利性组织&#xff0c;隶属于 Linux 基金会。CNCF 的主要目标是通过开源软件推动云原生计算技术的发展和普及&#xff0c;帮助企业更…

【C语言篇】C语言数据类型和变量

文章目录 C语言数据类型和变量1. 数据类型介绍1.1 字符型1.2 整形1.3 浮点型1.4 布尔类型1.5 各种类型数据长度1.5.1 sizeof操作符1.5.2 数据类型长度1.5.3 sizeof表达式不计算 2. signed和unsigned3. 数据类型的取值范围4. 变量4.1变量的创建4.2 变量的分类 5.强制类型转换 C语…

Redis:快速键值存储的入门指南

一、什么是Redis&#xff1f; Redis&#xff0c;全称为Remote Dictionary Server&#xff0c;是一种开源的、高性能的键值&#xff08;Key-Value&#xff09;存储系统。与传统的关系型数据库不同&#xff0c;Redis将数据主要存储在内存中&#xff0c;因此能够提供极低延迟的数…

网络传输层——UDP与TCP

前言&#xff1a; 1.国际网络体系结构&#xff1a; OSI模型: open system interconnect 理论模型 1977 国际标准化组织 各种不同体系结构的计算机能在世界范围内互联成网。 应用层:要传输的数据信息&#xff0c;如文件传输&#xff0c;电子邮件等…

【C语言】宏定义常量加 ; 的错误

我在使用宏定义常量定义二维数组的时候&#xff0c;编译器报错&#xff1a;应输入“]”&#xff0c;如下&#xff1a; 原因是宏定义不是C语言规定的语句&#xff0c;它的结尾不加 ; 。在上图的 int mine[EASY_ROWS][EASY_COLS]; 中&#xff0c;把 EASY_ROWS 替换为了 9;2; &…

Typora 以 Github 作为图床使用 PicGo 上传图片

本文简练快速介绍如标题所述的操作流程 文章目录 1.前言1.1 图床简述1.2 Github图床的优缺点1.2.1 优点1.2.2 缺点 2.下载PicGo3.Github访问加速4.用github创建图床服务器4.1 注册4.2 创建仓库 4.3 生成TOKEN令牌5.设置PicGo6.设置Typora7.完成 1.前言 1.1 图床简述 图床&…

车载音频记录

一、车载音频总线 2.8车载音频总线A2B Automotive Audio Bus_a2b总线-CSDN博客 传统的车载音频接口采用的是点对点模式&#xff0c;车身数字音频总线采用环型或者菊花链型总线。 A2B&#xff1a;Automotive Audio Bus 支持串联拓扑&#xff0c;即单个主机最多连接10个…

RedHat8安装Oracle19C

RedHat8安装Oracle19C 1、 更新yum源 更新yum源为阿里云镜像源&#xff1a; # 进入源目录 cd /etc/yum.repos.d/ # 删除 redhat 默认源 rm redhat.repo # 下载阿里云的centos7源 curl -O http://mirrors.aliyun.com/repo/Centos-8.repo # 替换 Centos-8.repo 中的 $releasev…

机器学习(二十二):精度和召回率

一、倾斜数据集 倾斜数据集&#xff1a;一个数据集中的正面和负面例子的比例非常不平衡&#xff0c;比如数据集中&#xff0c;结果为1的占比20%&#xff0c;结果为0的占比80% 例子&#xff1a;如果数据集的结果中只有0.5%是1&#xff0c;其余结果是0。有一个模型的预测准确度…

24.7.17数据结构|顺序表

目录 大O的工程意义&#xff1f; 线性表 引入&#xff1a; 主要掌握【代码实现】&#xff1a; 一、线性结构 1、逻辑描述 2、顺序表 1、如何定义结构 1&#xff09;静态顺序表 1&#xff09;动态顺序表 2、写代码 &#xff08;1&#xff09;【clion创建工程】 ​编…