docs: add code playground (#220)

improve-workflow
Meng Zhang 2023-06-07 13:05:47 -07:00 committed by GitHub
parent e7f5d38389
commit fd2adcba8a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 256 additions and 12 deletions

View File

@ -81,6 +81,7 @@ const config = {
position: 'left',
label: 'Docs',
},
{to: '/playground', label: 'Playground', position: 'left'},
{to: '/api', label: 'API', position: 'left'},
// FIXME(meng): enable blog.
// {to: '/blog', label: 'Blog', position: 'left'},
@ -97,11 +98,19 @@ const config = {
style: 'dark',
links: [
{
title: 'Docs',
title: 'Links',
items: [
{
label: 'Self Hosting',
to: '/docs/self-hosting',
label: 'Docs',
to: '/docs/getting-started',
},
{
label: 'Playground',
to: '/playground',
},
{
label: 'API',
to: '/api',
},
],
},

View File

@ -17,7 +17,9 @@
"@docusaurus/core": "2.4.1",
"@docusaurus/preset-classic": "2.4.1",
"@mdx-js/react": "^1.6.22",
"@monaco-editor/react": "^4.5.1",
"autoprefixer": "^10.4.14",
"axios": "^1.4.0",
"clsx": "^1.2.1",
"docusaurus-preset-openapi": "^0.6.4",
"postcss": "^8.4.24",

View File

@ -0,0 +1,185 @@
import axios from "axios"
import React, { useRef, useEffect } from "react"
import Editor, { useMonaco } from "@monaco-editor/react"
let TabbyServerURL = "https://app.tabbyml.com/api/workspace/tabbyml/tabby";
export default function MonacoEditor(props) {
function beforeMount(monaco) {
// Setup theming
const LIGHT_BACKGROUND = getColor(
"--ifm-background-color"
);
monaco.editor.defineTheme("tabby-light", {
base: "vs",
inherit: true,
rules: [],
colors: {
"editor.background": LIGHT_BACKGROUND,
}
});
monaco.languages.registerInlineCompletionsProvider(
{ pattern: "**" },
new CompletionProvider(monaco)
)
monaco.editor.registerCommand(
"acceptTabbyCompletion",
(accessor, id, index) => {
logAction(id, index, "select")
}
)
}
return (
<Editor
beforeMount={beforeMount}
theme="tabby-light"
defaultLanguage="python"
{...props}
/>
)
}
class CompletionProvider {
constructor(monaco) {
this.monaco = monaco
this.latestTimestamp = 0
}
async provideInlineCompletions(document, position, context, token) {
const prompt = this.getPrompt(document, position)
const emptyResponse = Promise.resolve({ items: [] })
if (this.isNil(prompt)) {
console.debug("Prompt is empty, skipping")
return emptyResponse
}
const currentTimestamp = Date.now()
this.latestTimestamp = currentTimestamp
await this.sleep(500)
if (this.pendingRequest) await this.pendingRequest
if (currentTimestamp < this.latestTimestamp) {
return emptyResponse
}
let response
try {
response = await this.callTabbyApi(currentTimestamp, prompt)
} catch (err) {
console.error("error", err)
return emptyResponse
}
const hasSuffixParen = this.hasSuffixParen(document, position)
const replaceRange = hasSuffixParen
? new this.monaco.Range(
position.lineNumber,
position.column,
position.lineNumber,
position.column + 1
)
: new this.monaco.Range(
position.lineNumber,
position.column,
position.lineNumber,
position.column
)
const items = this.toInlineCompletions(response.data, replaceRange)
return Promise.resolve({ data: response.data, items })
}
handleItemDidShow(completions, item) {
logAction(completions.data.id, item.choice.index, "view")
}
freeInlineCompletions() {}
getPrompt(document, position) {
const firstLine = Math.max(position.lineNumber - 120, 0)
const range = new this.monaco.Range(
firstLine,
0,
position.lineNumber,
position.column
)
return document.getValueInRange(range)
}
isNil(value) {
return value === undefined || value === null || value.length === 0
}
sleep(milliseconds) {
return new Promise((r) => setTimeout(r, milliseconds))
}
async callTabbyApi(timestamp, prompt) {
const request = (this.pendingRequest = axios.post(
`${TabbyServerURL}/v1/completions`,
{
language: "python",
prompt,
}
))
const response = await request
this.pendingRequest = null
return response
}
toInlineCompletions(value, range) {
return (
value.choices.map((choice) => ({
range,
insertText: choice.text,
choice,
command: {
id: "acceptTabbyCompletion",
arguments: [value.id, choice.index],
},
})) || []
)
}
hasSuffixParen(document, position) {
const suffix = document.getValueInRange(
new this.monaco.Range(
position.lineNumber,
position.column,
position.lineNumber,
position.column + 1
)
)
return ")]}".indexOf(suffix) > -1
}
}
function logAction(completion_id, choice_index, type) {
axios.post(`${TabbyServerURL}/v1/events`, {
type,
completion_id,
choice_index,
})
}
function getColor(property) {
const styles = getComputedStyle(document.documentElement);
// Weird chrome bug, returns " #ffffff " instead of "#ffffff", see: https://github.com/cloud-annotations/docusaurus-openapi/issues/144
const color = styles.getPropertyValue(property).trim();
if (color.length === 4) {
// change hex short codes like "#fff" to "#ffffff"
// to fix: https://github.com/cloud-annotations/docusaurus-openapi/issues/183
let res = "#"; // prepend #
for (const c of color.substring(1)) {
res += c + c; // duplicate each char
}
return res;
}
return color;
}

View File

@ -18,17 +18,14 @@
--ifm-color-primary-lighter: #443f45;
--ifm-color-primary-lightest: #4d484e;
--ifm-background-color: #e9e1d4;
--ifm-background-surface-color: var(--ifm-background-color);
--ifm-background-color: #FFF8EC;
--ifm-background-surface-color: #E9E1D4;
--ifm-code-background: #FFF8EC;
--openapi-monaco-background-color-light: #FFF8EC !important;
--ifm-code-background: #FFFEFD;
--openapi-monaco-background-color-light: var(--ifm-code-background) !important;
--openapi-card-background-color: var(--ifm-code-background) !important;
}
.monaco-editor{
background-color: var(--openapi-monaco-background-color-light) !important;
}
.markdown a {
color: var(--ifm-color-success-darkest);
@ -56,3 +53,7 @@
height: 10px;
font-weight: 600;
}
.Playground .monaco-editor {
padding-top: 15px;
}

View File

@ -0,0 +1,33 @@
import React from 'react';
import Layout from '@theme/Layout';
import Monaco from "@site/src/components/Monaco";
export default function Home() {
return (
<Layout title="Playground">
<main className="flex-grow flex relative">
<div className="Playground absolute inset-0">
<Monaco
defaultValue={DEFAULT_CODE}
options={{
automaticLayout: true,
fontSize: 16,
}}
/>
</div>
</main>
</Layout>
);
}
const DEFAULT_CODE = `import datetime
def parse_expenses(expenses_string):
"""Parse the list of expenses and return the list of triples (date, value, currency).
Ignore lines starting with #.
Parse the date using datetime.
Example expenses_string:
2016-01-02 -34.01 USD
2016-01-03 2.59 DKK
2016-01-03 -2.72 EUR
"""`

View File

@ -1804,7 +1804,7 @@
dependencies:
state-local "^1.0.6"
"@monaco-editor/react@^4.3.1":
"@monaco-editor/react@^4.3.1", "@monaco-editor/react@^4.5.1":
version "4.5.1"
resolved "https://registry.yarnpkg.com/@monaco-editor/react/-/react-4.5.1.tgz#fbc76c692aee9a33b9ab24ae0c5f219b8f002fdb"
integrity sha512-NNDFdP+2HojtNhCkRfE6/D6ro6pBNihaOzMbGK84lNWzRu+CfBjwzGt4jmnqimLuqp5yE5viHS2vi+QOAnD5FQ==
@ -2681,6 +2681,15 @@ axios@^0.26.1:
dependencies:
follow-redirects "^1.14.8"
axios@^1.4.0:
version "1.4.0"
resolved "https://registry.yarnpkg.com/axios/-/axios-1.4.0.tgz#38a7bf1224cd308de271146038b551d725f0be1f"
integrity sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==
dependencies:
follow-redirects "^1.15.0"
form-data "^4.0.0"
proxy-from-env "^1.1.0"
babel-loader@^8.2.5:
version "8.3.0"
resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.3.0.tgz#124936e841ba4fe8176786d6ff28add1f134d6a8"
@ -4257,7 +4266,7 @@ flux@^4.0.1:
fbemitter "^3.0.0"
fbjs "^3.0.1"
follow-redirects@^1.0.0, follow-redirects@^1.14.7, follow-redirects@^1.14.8:
follow-redirects@^1.0.0, follow-redirects@^1.14.7, follow-redirects@^1.14.8, follow-redirects@^1.15.0:
version "1.15.2"
resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.2.tgz#b460864144ba63f2681096f274c4e57026da2c13"
integrity sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==
@ -6781,6 +6790,11 @@ proxy-addr@~2.0.7:
forwarded "0.2.0"
ipaddr.js "1.9.1"
proxy-from-env@^1.1.0:
version "1.1.0"
resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2"
integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==
pump@^3.0.0:
version "3.0.0"
resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64"