Upload folder using huggingface_hub
Browse filesThis view is limited to 50 files because it contains too many changes.
See raw diff
- .changeset/README.md +8 -0
- .changeset/afraid-papayas-own.md +5 -0
- .changeset/changeset.cjs +332 -0
- .changeset/config.json +11 -0
- .changeset/fix_changelogs.cjs +149 -0
- .changeset/silly-hairs-burn.md +5 -0
- .changeset/stupid-snakes-build.md +6 -0
- .config/.prettierignore +35 -0
- .config/.prettierrc.json +8 -0
- .config/basevite.config.ts +93 -0
- .config/copy_frontend.py +63 -0
- .config/demos.json +34 -0
- .config/eslint.config.js +164 -0
- .config/lite-builder/pyproject.toml +33 -0
- .config/lite-builder/src/lite_builder/__init__.py +0 -0
- .config/lite-builder/src/lite_builder/builder.py +5 -0
- .config/lite-builder/src/lite_builder/hooks.py +6 -0
- .config/playwright-ct.config.ts +41 -0
- .config/playwright-setup.js +179 -0
- .config/playwright.config.js +73 -0
- .config/playwright/index.html +12 -0
- .config/playwright/index.ts +2 -0
- .config/postcss.config.cjs +8 -0
- .config/setup_vite_tests.ts +8 -0
- .config/svelte.config.js +5 -0
- .config/tailwind.config.cjs +12 -0
- .config/vitest.config.ts +3 -0
- .devcontainer/devcontainer.json +41 -0
- .dockerignore +41 -0
- .editorconfig +8 -0
- .git-blame-ignore-revs +14 -0
- .gitattributes +35 -0
- .github/ISSUE_TEMPLATE/bug_report_template.yml +69 -0
- .github/ISSUE_TEMPLATE/config.yml +5 -0
- .github/ISSUE_TEMPLATE/feature_request.md +19 -0
- .github/PULL_REQUEST_TEMPLATE.md +18 -0
- .github/actions/changes/action.yml +78 -0
- .github/actions/install-all-deps/action.yml +92 -0
- .github/actions/install-frontend-deps/action.yml +51 -0
- .github/configs/semgrep_rules.yaml +110 -0
- .github/filters.json +51 -0
- .github/stale +17 -0
- .github/workflows/comment-queue.yml +37 -0
- .github/workflows/delete-stale-spaces.yml +42 -0
- .github/workflows/generate-changeset.yml +96 -0
- .github/workflows/npm-previews.yml +53 -0
- .github/workflows/previews-build.yml +90 -0
- .github/workflows/previews-deploy.yml +176 -0
- .github/workflows/publish.yml +80 -0
- .github/workflows/semgrep.yml +68 -0
.changeset/README.md
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Changesets
|
2 |
+
|
3 |
+
Hello and welcome! This folder has been automatically generated by `@changesets/cli`, a build tool that works
|
4 |
+
with multi-package repos, or single-package repos to help you version and publish your code. You can
|
5 |
+
find the full documentation for it [in our repository](https://github.com/changesets/changesets)
|
6 |
+
|
7 |
+
We have a quick list of common questions to get you started engaging with this project in
|
8 |
+
[our documentation](https://github.com/changesets/changesets/blob/main/docs/common-questions.md)
|
.changeset/afraid-papayas-own.md
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
---
|
2 |
+
"gradio": patch
|
3 |
+
---
|
4 |
+
|
5 |
+
feat:Tweak message shown in Colab notebooks
|
.changeset/changeset.cjs
ADDED
@@ -0,0 +1,332 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
const { getPackagesSync } = require("@manypkg/get-packages");
|
2 |
+
const dependents_graph = require("@changesets/get-dependents-graph");
|
3 |
+
|
4 |
+
const gh = require("@changesets/get-github-info");
|
5 |
+
const { existsSync, readFileSync, writeFileSync } = require("fs");
|
6 |
+
const { join } = require("path");
|
7 |
+
|
8 |
+
const { getInfo, getInfoFromPullRequest } = gh;
|
9 |
+
const pkg_data = getPackagesSync(process.cwd());
|
10 |
+
const { packages, rootDir } = pkg_data;
|
11 |
+
const dependents = dependents_graph.getDependentsGraph({
|
12 |
+
packages,
|
13 |
+
root: pkg_data.rootPackage
|
14 |
+
});
|
15 |
+
|
16 |
+
/**
|
17 |
+
* @typedef {{packageJson: {name: string, python?: boolean}, dir: string}} Package
|
18 |
+
*/
|
19 |
+
|
20 |
+
/**
|
21 |
+
* @typedef {{summary: string, id: string, commit: string, releases: {name: string}}} Changeset
|
22 |
+
*/
|
23 |
+
|
24 |
+
/**
|
25 |
+
*
|
26 |
+
* @param {string} package_name The name of the package to find the directories for
|
27 |
+
* @returns {string[]} The directories for the package
|
28 |
+
*/
|
29 |
+
function find_packages_dirs(package_name) {
|
30 |
+
/** @type {string[]} */
|
31 |
+
let package_dirs = [];
|
32 |
+
|
33 |
+
/** @type {Package | undefined} */
|
34 |
+
const _package = packages.find((p) => p.packageJson.name === package_name);
|
35 |
+
if (!_package) throw new Error(`Package ${package_name} not found`);
|
36 |
+
|
37 |
+
package_dirs.push(_package.dir);
|
38 |
+
if (_package.packageJson.python) {
|
39 |
+
package_dirs.push(join(_package.dir, ".."));
|
40 |
+
}
|
41 |
+
return package_dirs;
|
42 |
+
}
|
43 |
+
|
44 |
+
let lines = {
|
45 |
+
_handled: []
|
46 |
+
};
|
47 |
+
|
48 |
+
const changelogFunctions = {
|
49 |
+
/**
|
50 |
+
*
|
51 |
+
* @param {Changeset[]} changesets The changesets that have been created
|
52 |
+
* @param {any} dependenciesUpdated The dependencies that have been updated
|
53 |
+
* @param {any} options The options passed to the changelog generator
|
54 |
+
* @returns {Promise<string>} The release line for the dependencies
|
55 |
+
*/
|
56 |
+
getDependencyReleaseLine: async (
|
57 |
+
changesets,
|
58 |
+
dependenciesUpdated,
|
59 |
+
options
|
60 |
+
) => {
|
61 |
+
if (!options.repo) {
|
62 |
+
throw new Error(
|
63 |
+
'Please provide a repo to this changelog generator like this:\n"changelog": ["@changesets/changelog-github", { "repo": "org/repo" }]'
|
64 |
+
);
|
65 |
+
}
|
66 |
+
if (dependenciesUpdated.length === 0) return "";
|
67 |
+
|
68 |
+
const changesetLink = `- Updated dependencies [${(
|
69 |
+
await Promise.all(
|
70 |
+
changesets.map(async (cs) => {
|
71 |
+
if (cs.commit) {
|
72 |
+
let { links } = await getInfo({
|
73 |
+
repo: options.repo,
|
74 |
+
commit: cs.commit
|
75 |
+
});
|
76 |
+
return links.commit;
|
77 |
+
}
|
78 |
+
})
|
79 |
+
)
|
80 |
+
)
|
81 |
+
.filter((_) => _)
|
82 |
+
.join(", ")}]:`;
|
83 |
+
|
84 |
+
const updatedDepenenciesList = dependenciesUpdated.map(
|
85 |
+
/**
|
86 |
+
*
|
87 |
+
* @param {any} dependency The dependency that has been updated
|
88 |
+
* @returns {string} The formatted dependency
|
89 |
+
*/
|
90 |
+
(dependency) => {
|
91 |
+
const updates = dependents.get(dependency.name);
|
92 |
+
|
93 |
+
if (updates && updates.length > 0) {
|
94 |
+
updates.forEach((update) => {
|
95 |
+
if (!lines[update]) {
|
96 |
+
lines[update] = {
|
97 |
+
dirs: find_packages_dirs(update),
|
98 |
+
current_changelog: "",
|
99 |
+
feat: [],
|
100 |
+
fix: [],
|
101 |
+
highlight: [],
|
102 |
+
previous_version: packages.find(
|
103 |
+
(p) => p.packageJson.name === update
|
104 |
+
).packageJson.version,
|
105 |
+
dependencies: []
|
106 |
+
};
|
107 |
+
|
108 |
+
const changelog_path = join(
|
109 |
+
//@ts-ignore
|
110 |
+
lines[update].dirs[1] || lines[update].dirs[0],
|
111 |
+
"CHANGELOG.md"
|
112 |
+
);
|
113 |
+
|
114 |
+
if (existsSync(changelog_path)) {
|
115 |
+
//@ts-ignore
|
116 |
+
lines[update].current_changelog = readFileSync(
|
117 |
+
changelog_path,
|
118 |
+
"utf-8"
|
119 |
+
)
|
120 |
+
.replace(`# ${update}`, "")
|
121 |
+
.trim();
|
122 |
+
}
|
123 |
+
}
|
124 |
+
lines[update].dependencies.push(
|
125 |
+
` - ${dependency.name}@${dependency.newVersion}`
|
126 |
+
);
|
127 |
+
});
|
128 |
+
}
|
129 |
+
|
130 |
+
return ` - ${dependency.name}@${dependency.newVersion}`;
|
131 |
+
}
|
132 |
+
);
|
133 |
+
|
134 |
+
writeFileSync(
|
135 |
+
join(rootDir, ".changeset", "_changelog.json"),
|
136 |
+
JSON.stringify(lines, null, 2)
|
137 |
+
);
|
138 |
+
|
139 |
+
return [changesetLink, ...updatedDepenenciesList].join("\n");
|
140 |
+
},
|
141 |
+
/**
|
142 |
+
*
|
143 |
+
* @param {{summary: string, id: string, commit: string, releases: {name: string}[]}} changeset The changeset that has been created
|
144 |
+
* @param {any} type The type of changeset
|
145 |
+
* @param {any} options The options passed to the changelog generator
|
146 |
+
* @returns {Promise<string>} The release line for the changeset
|
147 |
+
*/
|
148 |
+
getReleaseLine: async (changeset, type, options) => {
|
149 |
+
if (!options || !options.repo) {
|
150 |
+
throw new Error(
|
151 |
+
'Please provide a repo to this changelog generator like this:\n"changelog": ["@changesets/changelog-github", { "repo": "org/repo" }]'
|
152 |
+
);
|
153 |
+
}
|
154 |
+
|
155 |
+
let prFromSummary;
|
156 |
+
let commitFromSummary;
|
157 |
+
/**
|
158 |
+
* @type {string[]}
|
159 |
+
*/
|
160 |
+
let usersFromSummary = [];
|
161 |
+
|
162 |
+
const replacedChangelog = changeset.summary
|
163 |
+
.replace(/^\s*(?:pr|pull|pull\s+request):\s*#?(\d+)/im, (_, pr) => {
|
164 |
+
let num = Number(pr);
|
165 |
+
if (!isNaN(num)) prFromSummary = num;
|
166 |
+
return "";
|
167 |
+
})
|
168 |
+
.replace(/^\s*commit:\s*([^\s]+)/im, (_, commit) => {
|
169 |
+
commitFromSummary = commit;
|
170 |
+
return "";
|
171 |
+
})
|
172 |
+
.replace(/^\s*(?:author|user):\s*@?([^\s]+)/gim, (_, user) => {
|
173 |
+
usersFromSummary.push(user);
|
174 |
+
return "";
|
175 |
+
})
|
176 |
+
.trim();
|
177 |
+
|
178 |
+
const [firstLine, ...futureLines] = replacedChangelog
|
179 |
+
.split("\n")
|
180 |
+
.map((l) => l.trimRight());
|
181 |
+
|
182 |
+
const links = await (async () => {
|
183 |
+
if (prFromSummary !== undefined) {
|
184 |
+
let { links } = await getInfoFromPullRequest({
|
185 |
+
repo: options.repo,
|
186 |
+
pull: prFromSummary
|
187 |
+
});
|
188 |
+
if (commitFromSummary) {
|
189 |
+
links = {
|
190 |
+
...links,
|
191 |
+
commit: `[\`${commitFromSummary}\`](https://github.com/${options.repo}/commit/${commitFromSummary})`
|
192 |
+
};
|
193 |
+
}
|
194 |
+
return links;
|
195 |
+
}
|
196 |
+
const commitToFetchFrom = commitFromSummary || changeset.commit;
|
197 |
+
if (commitToFetchFrom) {
|
198 |
+
let { links } = await getInfo({
|
199 |
+
repo: options.repo,
|
200 |
+
commit: commitToFetchFrom
|
201 |
+
});
|
202 |
+
return links;
|
203 |
+
}
|
204 |
+
return {
|
205 |
+
commit: null,
|
206 |
+
pull: null,
|
207 |
+
user: null
|
208 |
+
};
|
209 |
+
})();
|
210 |
+
|
211 |
+
const user_link = /\[(@[^]+)\]/.exec(links.user);
|
212 |
+
const users =
|
213 |
+
usersFromSummary && usersFromSummary.length
|
214 |
+
? usersFromSummary
|
215 |
+
.map((userFromSummary) => `@${userFromSummary}`)
|
216 |
+
.join(", ")
|
217 |
+
: user_link
|
218 |
+
? user_link[1]
|
219 |
+
: links.user;
|
220 |
+
|
221 |
+
const prefix = [
|
222 |
+
links.pull === null ? "" : `${links.pull}`,
|
223 |
+
links.commit === null ? "" : `${links.commit}`
|
224 |
+
]
|
225 |
+
.join(" ")
|
226 |
+
.trim();
|
227 |
+
|
228 |
+
const suffix = users === null ? "" : ` Thanks ${users}!`;
|
229 |
+
|
230 |
+
/**
|
231 |
+
* @typedef {{[key: string]: string[] | {dirs: string[], current_changelog: string, feat: {summary: string}[], fix: {summary: string}[], highlight: {summary: string}[]}}} ChangesetMeta
|
232 |
+
*/
|
233 |
+
|
234 |
+
/**
|
235 |
+
* @type { ChangesetMeta & { _handled: string[] } }}
|
236 |
+
*/
|
237 |
+
|
238 |
+
if (lines._handled.includes(changeset.id)) {
|
239 |
+
return "done";
|
240 |
+
}
|
241 |
+
lines._handled.push(changeset.id);
|
242 |
+
|
243 |
+
changeset.releases.forEach((release) => {
|
244 |
+
if (!lines[release.name]) {
|
245 |
+
lines[release.name] = {
|
246 |
+
dirs: find_packages_dirs(release.name),
|
247 |
+
current_changelog: "",
|
248 |
+
feat: [],
|
249 |
+
fix: [],
|
250 |
+
highlight: [],
|
251 |
+
previous_version: packages.find(
|
252 |
+
(p) => p.packageJson.name === release.name
|
253 |
+
).packageJson.version,
|
254 |
+
dependencies: []
|
255 |
+
};
|
256 |
+
}
|
257 |
+
|
258 |
+
const changelog_path = join(
|
259 |
+
//@ts-ignore
|
260 |
+
lines[release.name].dirs[1] || lines[release.name].dirs[0],
|
261 |
+
"CHANGELOG.md"
|
262 |
+
);
|
263 |
+
|
264 |
+
if (existsSync(changelog_path)) {
|
265 |
+
//@ts-ignore
|
266 |
+
lines[release.name].current_changelog = readFileSync(
|
267 |
+
changelog_path,
|
268 |
+
"utf-8"
|
269 |
+
)
|
270 |
+
.replace(`# ${release.name}`, "")
|
271 |
+
.trim();
|
272 |
+
}
|
273 |
+
|
274 |
+
const [, _type, summary] = changeset.summary
|
275 |
+
.trim()
|
276 |
+
.match(/^(feat|fix|highlight)\s*:\s*([^]*)/im) || [
|
277 |
+
,
|
278 |
+
"feat",
|
279 |
+
changeset.summary
|
280 |
+
];
|
281 |
+
|
282 |
+
let formatted_summary = "";
|
283 |
+
|
284 |
+
if (_type === "highlight") {
|
285 |
+
const [heading, ...rest] = summary.trim().split("\n");
|
286 |
+
const _heading = `${heading} ${prefix ? `(${prefix})` : ""}`;
|
287 |
+
const _rest = rest.concat(["", suffix]);
|
288 |
+
|
289 |
+
formatted_summary = `${_heading}\n${_rest.join("\n")}`;
|
290 |
+
} else {
|
291 |
+
formatted_summary = handle_line(summary, prefix, suffix);
|
292 |
+
}
|
293 |
+
|
294 |
+
//@ts-ignore
|
295 |
+
lines[release.name][_type].push({
|
296 |
+
summary: formatted_summary
|
297 |
+
});
|
298 |
+
});
|
299 |
+
|
300 |
+
writeFileSync(
|
301 |
+
join(rootDir, ".changeset", "_changelog.json"),
|
302 |
+
JSON.stringify(lines, null, 2)
|
303 |
+
);
|
304 |
+
|
305 |
+
return `\n\n-${prefix ? `${prefix} -` : ""} ${firstLine}\n${futureLines
|
306 |
+
.map((l) => ` ${l}`)
|
307 |
+
.join("\n")}`;
|
308 |
+
}
|
309 |
+
};
|
310 |
+
|
311 |
+
/**
|
312 |
+
* @param {string} str The changelog entry
|
313 |
+
* @param {string} prefix The prefix to add to the first line
|
314 |
+
* @param {string} suffix The suffix to add to the last line
|
315 |
+
* @returns {string} The formatted changelog entry
|
316 |
+
*/
|
317 |
+
function handle_line(str, prefix, suffix) {
|
318 |
+
const [_s, ...lines] = str.split("\n").filter(Boolean);
|
319 |
+
|
320 |
+
const desc = `${prefix ? `${prefix} -` : ""} ${_s.replace(
|
321 |
+
/[\s\.]$/,
|
322 |
+
""
|
323 |
+
)}. ${suffix}`;
|
324 |
+
|
325 |
+
if (_s.length === 1) {
|
326 |
+
return desc;
|
327 |
+
}
|
328 |
+
|
329 |
+
return [desc, ...lines.map((l) => ` ${l}`)].join("/n");
|
330 |
+
}
|
331 |
+
|
332 |
+
module.exports = changelogFunctions;
|
.changeset/config.json
ADDED
@@ -0,0 +1,11 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"$schema": "https://unpkg.com/@changesets/config@2.3.0/schema.json",
|
3 |
+
"changelog": ["./changeset.cjs", { "repo": "gradio-app/gradio" }],
|
4 |
+
"commit": false,
|
5 |
+
"fixed": [],
|
6 |
+
"linked": [],
|
7 |
+
"access": "public",
|
8 |
+
"baseBranch": "main",
|
9 |
+
"updateInternalDependencies": "patch",
|
10 |
+
"ignore": ["@self/spaces-test", "@self/cdn-test"]
|
11 |
+
}
|
.changeset/fix_changelogs.cjs
ADDED
@@ -0,0 +1,149 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
const { join } = require("path");
|
2 |
+
const { readFileSync, existsSync, writeFileSync, unlinkSync } = require("fs");
|
3 |
+
const { getPackagesSync } = require("@manypkg/get-packages");
|
4 |
+
|
5 |
+
const RE_PKG_NAME = /^[\w-]+\b/;
|
6 |
+
const pkg_meta = getPackagesSync(process.cwd());
|
7 |
+
|
8 |
+
/**
|
9 |
+
* @typedef {{dirs: string[], highlight: {summary: string}[], feat: {summary: string}[], fix: {summary: string}[], current_changelog: string}} ChangesetMeta
|
10 |
+
*/
|
11 |
+
|
12 |
+
/**
|
13 |
+
* @typedef {{[key: string]: ChangesetMeta}} ChangesetMetaCollection
|
14 |
+
*/
|
15 |
+
|
16 |
+
function run() {
|
17 |
+
if (!existsSync(join(pkg_meta.rootDir, ".changeset", "_changelog.json"))) {
|
18 |
+
console.warn("No changesets to process");
|
19 |
+
return;
|
20 |
+
}
|
21 |
+
|
22 |
+
/**
|
23 |
+
* @type { ChangesetMetaCollection & { _handled: string[] } }}
|
24 |
+
*/
|
25 |
+
const { _handled, ...packages } = JSON.parse(
|
26 |
+
readFileSync(
|
27 |
+
join(pkg_meta.rootDir, ".changeset", "_changelog.json"),
|
28 |
+
"utf-8"
|
29 |
+
)
|
30 |
+
);
|
31 |
+
|
32 |
+
/**
|
33 |
+
* @typedef { {packageJson: {name: string, version: string, python: boolean}, dir: string} } PackageMeta
|
34 |
+
*/
|
35 |
+
|
36 |
+
/**
|
37 |
+
* @type { {[key:string]: PackageMeta} }
|
38 |
+
*/
|
39 |
+
const all_packages = pkg_meta.packages.reduce((acc, pkg) => {
|
40 |
+
acc[pkg.packageJson.name] = /**@type {PackageMeta} */ (
|
41 |
+
/** @type {unknown} */ (pkg)
|
42 |
+
);
|
43 |
+
return acc;
|
44 |
+
}, /** @type {{[key:string] : PackageMeta}} */ ({}));
|
45 |
+
|
46 |
+
for (const pkg_name in packages) {
|
47 |
+
const { dirs, highlight, feat, fix, current_changelog, dependencies } =
|
48 |
+
/**@type {ChangesetMeta} */ (packages[pkg_name]);
|
49 |
+
|
50 |
+
if (pkg_name === "@gradio/lite") {
|
51 |
+
const target = all_packages.gradio.packageJson.version.split(".");
|
52 |
+
|
53 |
+
const current_version = packages[pkg_name].previous_version.split(".");
|
54 |
+
|
55 |
+
if (!packages.gradio) {
|
56 |
+
const patch = parseInt(current_version[2]) + 1;
|
57 |
+
const new_version = [target[0], target[1], patch];
|
58 |
+
all_packages[pkg_name].packageJson.version = new_version.join(".");
|
59 |
+
} else {
|
60 |
+
if (parseInt(target[1]) > parseInt(current_version[1])) {
|
61 |
+
all_packages[pkg_name].packageJson.version = target.join(".");
|
62 |
+
} else if (parseInt(target[1]) === parseInt(current_version[1])) {
|
63 |
+
const patch = parseInt(current_version[2]) + 1;
|
64 |
+
const new_version = [target[0], target[1], patch];
|
65 |
+
all_packages[pkg_name].packageJson.version = new_version.join(".");
|
66 |
+
}
|
67 |
+
}
|
68 |
+
|
69 |
+
writeFileSync(
|
70 |
+
join(all_packages[pkg_name].dir, "package.json"),
|
71 |
+
JSON.stringify(all_packages[pkg_name].packageJson, null, "\t") + "\n"
|
72 |
+
);
|
73 |
+
}
|
74 |
+
|
75 |
+
const { version, python } = all_packages[pkg_name].packageJson;
|
76 |
+
|
77 |
+
const highlights = highlight?.map((h) => `${h.summary}`) || [];
|
78 |
+
const features = feat?.map((f) => `- ${f.summary}`) || [];
|
79 |
+
const fixes = fix?.map((f) => `- ${f.summary}`) || [];
|
80 |
+
const deps = Array.from(new Set(dependencies?.map((d) => d.trim()))) || [];
|
81 |
+
|
82 |
+
const release_notes = /** @type {[string[], string][]} */ ([
|
83 |
+
[highlights, "### Highlights"],
|
84 |
+
[features, "### Features"],
|
85 |
+
[fixes, "### Fixes"],
|
86 |
+
[deps, "### Dependency updates"]
|
87 |
+
])
|
88 |
+
.filter(([s], i) => s.length > 0)
|
89 |
+
.map(([lines, title]) => {
|
90 |
+
if (title === "### Highlights") {
|
91 |
+
return `${title}\n\n${lines.join("\n\n")}`;
|
92 |
+
}
|
93 |
+
|
94 |
+
return `${title}\n\n${lines.join("\n")}`;
|
95 |
+
})
|
96 |
+
.join("\n\n");
|
97 |
+
|
98 |
+
const new_changelog = `# ${pkg_name}
|
99 |
+
|
100 |
+
## ${version}
|
101 |
+
|
102 |
+
${release_notes}
|
103 |
+
|
104 |
+
${current_changelog.replace(`# ${pkg_name}`, "").trim()}
|
105 |
+
`.trim();
|
106 |
+
|
107 |
+
dirs.forEach((dir) => {
|
108 |
+
writeFileSync(join(dir, "CHANGELOG.md"), new_changelog);
|
109 |
+
});
|
110 |
+
|
111 |
+
if (python) {
|
112 |
+
bump_local_dependents(pkg_name, version);
|
113 |
+
}
|
114 |
+
}
|
115 |
+
|
116 |
+
unlinkSync(join(pkg_meta.rootDir, ".changeset", "_changelog.json"));
|
117 |
+
|
118 |
+
/**
|
119 |
+
* @param {string} pkg_to_bump The name of the package to bump
|
120 |
+
* @param {string} version The version to bump to
|
121 |
+
* @returns {void}
|
122 |
+
* */
|
123 |
+
function bump_local_dependents(pkg_to_bump, version) {
|
124 |
+
for (const pkg_name in all_packages) {
|
125 |
+
const {
|
126 |
+
dir,
|
127 |
+
packageJson: { python }
|
128 |
+
} = all_packages[pkg_name];
|
129 |
+
|
130 |
+
if (!python) continue;
|
131 |
+
|
132 |
+
const requirements_path = join(dir, "..", "requirements.txt");
|
133 |
+
const requirements = readFileSync(requirements_path, "utf-8").split("\n");
|
134 |
+
|
135 |
+
const pkg_index = requirements.findIndex((line) => {
|
136 |
+
const m = line.trim().match(RE_PKG_NAME);
|
137 |
+
if (!m) return false;
|
138 |
+
return m[0] === pkg_to_bump;
|
139 |
+
});
|
140 |
+
|
141 |
+
if (pkg_index !== -1) {
|
142 |
+
requirements[pkg_index] = `${pkg_to_bump}==${version}`;
|
143 |
+
writeFileSync(requirements_path, requirements.join("\n"));
|
144 |
+
}
|
145 |
+
}
|
146 |
+
}
|
147 |
+
}
|
148 |
+
|
149 |
+
run();
|
.changeset/silly-hairs-burn.md
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
---
|
2 |
+
"gradio": patch
|
3 |
+
---
|
4 |
+
|
5 |
+
feat:Fix streaming Audio/Video Output
|
.changeset/stupid-snakes-build.md
ADDED
@@ -0,0 +1,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
---
|
2 |
+
"@gradio/chatbot": patch
|
3 |
+
"gradio": patch
|
4 |
+
---
|
5 |
+
|
6 |
+
fix:Fix Audio in Chatbot bug
|
.config/.prettierignore
ADDED
@@ -0,0 +1,35 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
**/*.md
|
2 |
+
**/js/app/public/**
|
3 |
+
**/pnpm-workspace.yaml
|
4 |
+
**/js/app/dist/**
|
5 |
+
**/js/wasm/dist/**
|
6 |
+
**/js/preview/dist/**
|
7 |
+
**/client/js/dist/**
|
8 |
+
**/js/*/dist/**
|
9 |
+
**/pnpm-lock.yaml
|
10 |
+
**/js/plot/src/Plot.svelte
|
11 |
+
**/.svelte-kit/**
|
12 |
+
**/demo/**
|
13 |
+
**/gradio/**
|
14 |
+
**/.pnpm-store/**
|
15 |
+
**/.venv/**
|
16 |
+
|
17 |
+
/guides/**
|
18 |
+
**/.mypy_cache/**
|
19 |
+
!test-strategy.md
|
20 |
+
**/js/_space-test/**
|
21 |
+
../js/lite/src/theme.css
|
22 |
+
../js/storybook/theme.css
|
23 |
+
**/gradio_cached_examples/**
|
24 |
+
**/storybook-static/**
|
25 |
+
**/.vscode/**
|
26 |
+
sweep.yaml
|
27 |
+
**/.vercel/**
|
28 |
+
**/build/**
|
29 |
+
**/src/lib/json/**/*
|
30 |
+
**/playwright/.cache/**/*
|
31 |
+
**/theme/src/pollen.css
|
32 |
+
**/venv/**
|
33 |
+
../js/app/src/api_docs/CodeSnippet.svelte
|
34 |
+
../js/app/src/api_docs/RecordingSnippet.svelte
|
35 |
+
../.changeset/pre.json
|
.config/.prettierrc.json
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"useTabs": true,
|
3 |
+
"singleQuote": false,
|
4 |
+
"trailingComma": "none",
|
5 |
+
"printWidth": 80,
|
6 |
+
"plugins": ["prettier-plugin-svelte"],
|
7 |
+
"overrides": [{ "files": "*.svelte", "options": { "parser": "svelte" } }]
|
8 |
+
}
|
.config/basevite.config.ts
ADDED
@@ -0,0 +1,93 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { defineConfig } from "vite";
|
2 |
+
import { svelte } from "@sveltejs/vite-plugin-svelte";
|
3 |
+
import sveltePreprocess from "svelte-preprocess";
|
4 |
+
// @ts-ignore
|
5 |
+
import custom_media from "postcss-custom-media";
|
6 |
+
import global_data from "@csstools/postcss-global-data";
|
7 |
+
// @ts-ignore
|
8 |
+
import prefixer from "postcss-prefix-selector";
|
9 |
+
import { readFileSync } from "fs";
|
10 |
+
import { join } from "path";
|
11 |
+
import { fileURLToPath } from "url";
|
12 |
+
|
13 |
+
const __dirname = fileURLToPath(new URL(".", import.meta.url));
|
14 |
+
const version_path = join(__dirname, "..", "gradio", "package.json");
|
15 |
+
const theme_token_path = join(
|
16 |
+
__dirname,
|
17 |
+
"..",
|
18 |
+
"js",
|
19 |
+
"theme",
|
20 |
+
"src",
|
21 |
+
"tokens.css"
|
22 |
+
);
|
23 |
+
|
24 |
+
const version = JSON.parse(readFileSync(version_path, { encoding: "utf-8" }))
|
25 |
+
.version.trim()
|
26 |
+
.replace(/\./g, "-");
|
27 |
+
|
28 |
+
//@ts-ignore
|
29 |
+
export default defineConfig(({ mode }) => {
|
30 |
+
const production = mode === "production";
|
31 |
+
|
32 |
+
return {
|
33 |
+
server: {
|
34 |
+
port: 9876
|
35 |
+
},
|
36 |
+
|
37 |
+
build: {
|
38 |
+
sourcemap: false,
|
39 |
+
target: "esnext",
|
40 |
+
minify: production,
|
41 |
+
rollupOptions: {
|
42 |
+
external: ["virtual:component-loader"]
|
43 |
+
}
|
44 |
+
},
|
45 |
+
define: {
|
46 |
+
BUILD_MODE: production ? JSON.stringify("prod") : JSON.stringify("dev"),
|
47 |
+
BACKEND_URL: production
|
48 |
+
? JSON.stringify("")
|
49 |
+
: JSON.stringify("http://localhost:7860/"),
|
50 |
+
GRADIO_VERSION: JSON.stringify(version)
|
51 |
+
},
|
52 |
+
css: {
|
53 |
+
postcss: {
|
54 |
+
plugins: [
|
55 |
+
prefixer({
|
56 |
+
prefix: `.gradio-container-${version}`,
|
57 |
+
// @ts-ignore
|
58 |
+
transform(prefix, selector, prefixedSelector, fileName) {
|
59 |
+
if (selector.indexOf("gradio-container") > -1) {
|
60 |
+
return prefix;
|
61 |
+
} else if (
|
62 |
+
selector.indexOf(":root") > -1 ||
|
63 |
+
selector.indexOf("dark") > -1 ||
|
64 |
+
fileName.indexOf(".svelte") > -1
|
65 |
+
) {
|
66 |
+
return selector;
|
67 |
+
}
|
68 |
+
return prefixedSelector;
|
69 |
+
}
|
70 |
+
}),
|
71 |
+
custom_media()
|
72 |
+
]
|
73 |
+
}
|
74 |
+
},
|
75 |
+
plugins: [
|
76 |
+
svelte({
|
77 |
+
inspector: false,
|
78 |
+
compilerOptions: {
|
79 |
+
dev: !production
|
80 |
+
},
|
81 |
+
hot: !process.env.VITEST && !production,
|
82 |
+
preprocess: sveltePreprocess({
|
83 |
+
postcss: {
|
84 |
+
plugins: [
|
85 |
+
global_data({ files: [theme_token_path] }),
|
86 |
+
custom_media()
|
87 |
+
]
|
88 |
+
}
|
89 |
+
})
|
90 |
+
})
|
91 |
+
]
|
92 |
+
};
|
93 |
+
});
|
.config/copy_frontend.py
ADDED
@@ -0,0 +1,63 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from __future__ import annotations
|
2 |
+
|
3 |
+
import shutil
|
4 |
+
import pathlib
|
5 |
+
from typing import Any
|
6 |
+
|
7 |
+
from hatchling.builders.hooks.plugin.interface import BuildHookInterface
|
8 |
+
|
9 |
+
|
10 |
+
def copy_js_code(root: str | pathlib.Path):
|
11 |
+
NOT_COMPONENT = [
|
12 |
+
"app",
|
13 |
+
"node_modules",
|
14 |
+
"storybook",
|
15 |
+
"playwright-report",
|
16 |
+
"workbench",
|
17 |
+
"tooltils",
|
18 |
+
"component-test",
|
19 |
+
"core",
|
20 |
+
"spa",
|
21 |
+
]
|
22 |
+
for entry in (pathlib.Path(root) / "js").iterdir():
|
23 |
+
if (
|
24 |
+
entry.is_dir()
|
25 |
+
and not str(entry.name).startswith("_")
|
26 |
+
and not str(entry.name) in NOT_COMPONENT
|
27 |
+
):
|
28 |
+
|
29 |
+
def ignore(s, names):
|
30 |
+
ignored = []
|
31 |
+
for n in names:
|
32 |
+
if (
|
33 |
+
n.startswith("CHANGELOG")
|
34 |
+
or n.startswith("README.md")
|
35 |
+
or n.startswith("node_modules")
|
36 |
+
or ".test." in n
|
37 |
+
or ".stories." in n
|
38 |
+
or ".spec." in n
|
39 |
+
):
|
40 |
+
ignored.append(n)
|
41 |
+
return ignored
|
42 |
+
|
43 |
+
shutil.copytree(
|
44 |
+
str(entry),
|
45 |
+
str(pathlib.Path("gradio") / "_frontend_code" / entry.name),
|
46 |
+
ignore=ignore,
|
47 |
+
dirs_exist_ok=True,
|
48 |
+
)
|
49 |
+
shutil.copytree(
|
50 |
+
str(pathlib.Path(root) / "client" / "js"),
|
51 |
+
str(pathlib.Path("gradio") / "_frontend_code" / "client"),
|
52 |
+
ignore=lambda d, names: ["node_modules", "test"],
|
53 |
+
dirs_exist_ok=True,
|
54 |
+
)
|
55 |
+
|
56 |
+
|
57 |
+
class BuildHook(BuildHookInterface):
|
58 |
+
def initialize(self, version: str, build_data: dict[str, Any]) -> None:
|
59 |
+
copy_js_code(self.root)
|
60 |
+
|
61 |
+
|
62 |
+
if __name__ == "__main__":
|
63 |
+
copy_js_code(pathlib.Path("..").resolve())
|
.config/demos.json
ADDED
@@ -0,0 +1,34 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
[
|
2 |
+
"audio_debugger",
|
3 |
+
"blocks_essay",
|
4 |
+
"blocks_group",
|
5 |
+
"blocks_js_methods",
|
6 |
+
"blocks_layout",
|
7 |
+
"blocks_multiple_event_triggers",
|
8 |
+
"blocks_update",
|
9 |
+
"calculator",
|
10 |
+
"cancel_events",
|
11 |
+
"chatbot_multimodal",
|
12 |
+
"chatinterface_streaming_echo",
|
13 |
+
"clear_components",
|
14 |
+
"code",
|
15 |
+
"fake_gan",
|
16 |
+
"fake_diffusion_with_gif",
|
17 |
+
"file_explorer_component_events",
|
18 |
+
"image_mod_default_image",
|
19 |
+
"image_editor_events",
|
20 |
+
"image_segmentation",
|
21 |
+
"interface_random_slider",
|
22 |
+
"kitchen_sink",
|
23 |
+
"kitchen_sink_random",
|
24 |
+
"matrix_transpose",
|
25 |
+
"mini_leaderboard",
|
26 |
+
"model3D",
|
27 |
+
"native_plots",
|
28 |
+
"reverse_audio",
|
29 |
+
"stt_or_tts",
|
30 |
+
"stream_audio",
|
31 |
+
"stream_frames",
|
32 |
+
"video_component",
|
33 |
+
"zip_files"
|
34 |
+
]
|
.config/eslint.config.js
ADDED
@@ -0,0 +1,164 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import globals from "globals";
|
2 |
+
import ts_plugin from "@typescript-eslint/eslint-plugin";
|
3 |
+
import js_plugin from "@eslint/js";
|
4 |
+
import jsdoc from "eslint-plugin-jsdoc";
|
5 |
+
|
6 |
+
import typescriptParser from "@typescript-eslint/parser";
|
7 |
+
import sveltePlugin from "eslint-plugin-svelte";
|
8 |
+
import svelteParser from "svelte-eslint-parser";
|
9 |
+
|
10 |
+
const ts_rules_disabled = Object.fromEntries(
|
11 |
+
Object.keys(ts_plugin.rules).map((rule) => [
|
12 |
+
`@typescript-eslint/${rule}`,
|
13 |
+
"off"
|
14 |
+
])
|
15 |
+
);
|
16 |
+
const js_rules_disabled = Object.fromEntries(
|
17 |
+
Object.keys(js_plugin.configs.all.rules).map((rule) => [rule, "off"])
|
18 |
+
);
|
19 |
+
|
20 |
+
const jsdoc_rules_disabled = Object.fromEntries(
|
21 |
+
Object.keys(jsdoc.configs.recommended.rules).map((rule) => [
|
22 |
+
`jsdoc/${rule}`,
|
23 |
+
"off"
|
24 |
+
])
|
25 |
+
);
|
26 |
+
|
27 |
+
const js_rules = {
|
28 |
+
...js_rules_disabled,
|
29 |
+
"no-console": ["error", { allow: ["warn", "error", "debug", "info"] }],
|
30 |
+
"no-constant-condition": "error",
|
31 |
+
"no-dupe-args": "error",
|
32 |
+
"no-extra-boolean-cast": "error",
|
33 |
+
"no-unexpected-multiline": "error",
|
34 |
+
"no-unreachable": "error",
|
35 |
+
"array-callback-return": "error",
|
36 |
+
complexity: "error",
|
37 |
+
"no-else-return": "error",
|
38 |
+
"no-useless-return": "error",
|
39 |
+
"no-undef": "error"
|
40 |
+
};
|
41 |
+
|
42 |
+
const ts_rules = {
|
43 |
+
...ts_rules_disabled,
|
44 |
+
"@typescript-eslint/adjacent-overload-signatures": "error",
|
45 |
+
"@typescript-eslint/explicit-function-return-type": [
|
46 |
+
"error",
|
47 |
+
{ allowExpressions: true }
|
48 |
+
],
|
49 |
+
"@typescript-eslint/consistent-type-exports": "error",
|
50 |
+
"@typescript-eslint/ban-types": "error",
|
51 |
+
"@typescript-eslint/array-type": "error",
|
52 |
+
"@typescript-eslint/no-inferrable-types": "error"
|
53 |
+
};
|
54 |
+
|
55 |
+
const jsdoc_rules = {
|
56 |
+
...jsdoc_rules_disabled,
|
57 |
+
"jsdoc/require-param-description": "error",
|
58 |
+
"jsdoc/require-returns-description": "error"
|
59 |
+
};
|
60 |
+
|
61 |
+
const { browser, es2021, node } = globals;
|
62 |
+
|
63 |
+
export default [
|
64 |
+
{
|
65 |
+
ignores: [
|
66 |
+
"**/.svelte-kit/**/*",
|
67 |
+
"**/node_modules/**",
|
68 |
+
"**/dist/**",
|
69 |
+
"**/.config/*",
|
70 |
+
"**/*.spec.ts",
|
71 |
+
"**/*.test.ts",
|
72 |
+
"**/*.node-test.ts",
|
73 |
+
"js/spa/test/**/*",
|
74 |
+
"**/*vite.config.ts",
|
75 |
+
"**/_website/**/*",
|
76 |
+
"**/app/**/*",
|
77 |
+
"**/_spaces-test/**/*",
|
78 |
+
"**/preview/test/**/*",
|
79 |
+
"**/component-test/**/*",
|
80 |
+
"**/js/wasm/src/webworker/**/*"
|
81 |
+
]
|
82 |
+
},
|
83 |
+
{
|
84 |
+
files: ["**/*.js", "**/*.cjs"],
|
85 |
+
languageOptions: {
|
86 |
+
globals: {
|
87 |
+
...browser,
|
88 |
+
...es2021,
|
89 |
+
...node
|
90 |
+
}
|
91 |
+
},
|
92 |
+
|
93 |
+
plugins: {
|
94 |
+
"eslint:recommended": js_plugin,
|
95 |
+
jsdoc
|
96 |
+
},
|
97 |
+
rules: { ...js_rules, ...jsdoc_rules }
|
98 |
+
},
|
99 |
+
|
100 |
+
{
|
101 |
+
files: ["**/*.ts"],
|
102 |
+
languageOptions: {
|
103 |
+
parser: typescriptParser,
|
104 |
+
parserOptions: {
|
105 |
+
project: "./tsconfig.json",
|
106 |
+
extraFileExtensions: [".svelte"]
|
107 |
+
},
|
108 |
+
globals: {
|
109 |
+
...browser,
|
110 |
+
...es2021,
|
111 |
+
...node
|
112 |
+
}
|
113 |
+
},
|
114 |
+
|
115 |
+
plugins: {
|
116 |
+
"@typescript-eslint": ts_plugin,
|
117 |
+
"eslint:recommended": js_plugin,
|
118 |
+
jsdoc
|
119 |
+
},
|
120 |
+
rules: {
|
121 |
+
...ts_rules,
|
122 |
+
...js_rules,
|
123 |
+
...jsdoc_rules,
|
124 |
+
"no-undef": "off"
|
125 |
+
}
|
126 |
+
},
|
127 |
+
{
|
128 |
+
files: ["**/client/js/**"],
|
129 |
+
languageOptions: {
|
130 |
+
parserOptions: {
|
131 |
+
project: "./client/js/tsconfig.json"
|
132 |
+
}
|
133 |
+
}
|
134 |
+
},
|
135 |
+
{
|
136 |
+
files: ["**/*.svelte"],
|
137 |
+
languageOptions: {
|
138 |
+
parser: svelteParser,
|
139 |
+
parserOptions: {
|
140 |
+
parser: typescriptParser,
|
141 |
+
project: "./tsconfig.json",
|
142 |
+
extraFileExtensions: [".svelte"]
|
143 |
+
},
|
144 |
+
globals: {
|
145 |
+
...browser,
|
146 |
+
...es2021
|
147 |
+
}
|
148 |
+
},
|
149 |
+
plugins: {
|
150 |
+
svelte: sveltePlugin,
|
151 |
+
"@typescript-eslint": ts_plugin,
|
152 |
+
"eslint:recommended": js_plugin,
|
153 |
+
jsdoc
|
154 |
+
},
|
155 |
+
rules: {
|
156 |
+
...ts_rules,
|
157 |
+
...js_rules,
|
158 |
+
...jsdoc_rules,
|
159 |
+
...sveltePlugin.configs.recommended.rules,
|
160 |
+
"svelte/no-at-html-tags": "off",
|
161 |
+
"no-undef": "off"
|
162 |
+
}
|
163 |
+
}
|
164 |
+
];
|
.config/lite-builder/pyproject.toml
ADDED
@@ -0,0 +1,33 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
[build-system]
|
2 |
+
requires = ["hatchling",]
|
3 |
+
build-backend = "hatchling.build"
|
4 |
+
|
5 |
+
[project]
|
6 |
+
name = "lite-builder"
|
7 |
+
description = "Python library for easily interacting with trained machine learning models"
|
8 |
+
license = "Apache-2.0"
|
9 |
+
version = "0.0.2"
|
10 |
+
requires-python = ">=3.8"
|
11 |
+
authors = [
|
12 |
+
{ name = "Abubakar Abid", email = "gradio-team@huggingface.co" },
|
13 |
+
{ name = "Ali Abid", email = "gradio-team@huggingface.co" },
|
14 |
+
{ name = "Ali Abdalla", email = "gradio-team@huggingface.co" },
|
15 |
+
{ name = "Dawood Khan", email = "gradio-team@huggingface.co" },
|
16 |
+
{ name = "Ahsen Khaliq", email = "gradio-team@huggingface.co" },
|
17 |
+
{ name = "Pete Allen", email = "gradio-team@huggingface.co" },
|
18 |
+
{ name = "Ömer Faruk Özdemir", email = "gradio-team@huggingface.co" },
|
19 |
+
{ name = "Freddy A Boulton", email = "gradio-team@huggingface.co" },
|
20 |
+
{ name = "Hannah Blair", email = "gradio-team@huggingface.co" },
|
21 |
+
]
|
22 |
+
keywords = ["machine learning", "reproducibility", "visualization"]
|
23 |
+
|
24 |
+
classifiers = [
|
25 |
+
'Development Status :: 5 - Production/Stable',
|
26 |
+
]
|
27 |
+
|
28 |
+
[tool.hatch.build]
|
29 |
+
sources = ["src"]
|
30 |
+
only-packages = true
|
31 |
+
|
32 |
+
[project.entry-points.hatch]
|
33 |
+
lite_builder = "lite_builder.hooks"
|
.config/lite-builder/src/lite_builder/__init__.py
ADDED
File without changes
|
.config/lite-builder/src/lite_builder/builder.py
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from hatchling.builders.wheel import WheelBuilder
|
2 |
+
|
3 |
+
|
4 |
+
class LiteBuilder(WheelBuilder):
|
5 |
+
PLUGIN_NAME = 'lite'
|
.config/lite-builder/src/lite_builder/hooks.py
ADDED
@@ -0,0 +1,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from hatchling.plugin import hookimpl
|
2 |
+
from .builder import LiteBuilder
|
3 |
+
|
4 |
+
@hookimpl
|
5 |
+
def hatch_register_builder():
|
6 |
+
return LiteBuilder
|
.config/playwright-ct.config.ts
ADDED
@@ -0,0 +1,41 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { defineConfig, devices } from "@playwright/experimental-ct-svelte";
|
2 |
+
import config from "./basevite.config";
|
3 |
+
|
4 |
+
/**
|
5 |
+
* See https://playwright.dev/docs/test-configuration.
|
6 |
+
*/
|
7 |
+
export default defineConfig({
|
8 |
+
testDir: "../",
|
9 |
+
/* The base directory, relative to the config file, for snapshot files created with toMatchSnapshot and toHaveScreenshot. */
|
10 |
+
snapshotDir: "./__snapshots__",
|
11 |
+
/* Maximum time one test can run for. */
|
12 |
+
timeout: 10 * 1000,
|
13 |
+
/* Run tests in files in parallel */
|
14 |
+
fullyParallel: true,
|
15 |
+
/* Fail the build on CI if you accidentally left test.only in the source code. */
|
16 |
+
forbidOnly: !!process.env.CI,
|
17 |
+
/* Retry on CI only */
|
18 |
+
retries: process.env.CI ? 2 : 0,
|
19 |
+
/* Opt out of parallel tests on CI. */
|
20 |
+
workers: process.env.CI ? 1 : undefined,
|
21 |
+
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
22 |
+
reporter: "html",
|
23 |
+
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */
|
24 |
+
use: {
|
25 |
+
/* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */
|
26 |
+
trace: "on-first-retry",
|
27 |
+
|
28 |
+
/* Port to use for Playwright component endpoint. */
|
29 |
+
ctPort: 3100,
|
30 |
+
ctViteConfig: config({ mode: "development", command: "build" })
|
31 |
+
},
|
32 |
+
testMatch: "*.component.spec.ts",
|
33 |
+
|
34 |
+
/* Configure projects for major browsers */
|
35 |
+
projects: [
|
36 |
+
{
|
37 |
+
name: "chromium",
|
38 |
+
use: { ...devices["Desktop Chrome"] }
|
39 |
+
}
|
40 |
+
]
|
41 |
+
});
|
.config/playwright-setup.js
ADDED
@@ -0,0 +1,179 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { spawn } from "node:child_process";
|
2 |
+
import { join, basename } from "path";
|
3 |
+
import { fileURLToPath } from "url";
|
4 |
+
import { readdirSync, writeFileSync } from "fs";
|
5 |
+
import net from "net";
|
6 |
+
|
7 |
+
import kl from "kleur";
|
8 |
+
|
9 |
+
const __dirname = fileURLToPath(new URL(".", import.meta.url));
|
10 |
+
const TEST_APP_PATH = join(__dirname, "./test.py");
|
11 |
+
const TEST_FILES_PATH = join(__dirname, "..", "js", "spa", "test");
|
12 |
+
const ROOT = join(__dirname, "..");
|
13 |
+
|
14 |
+
const test_files = readdirSync(TEST_FILES_PATH)
|
15 |
+
.filter(
|
16 |
+
(f) =>
|
17 |
+
f.endsWith("spec.ts") &&
|
18 |
+
!f.endsWith(".skip.spec.ts") &&
|
19 |
+
!f.endsWith(".component.spec.ts") &&
|
20 |
+
!f.endsWith(".reload.spec.ts")
|
21 |
+
)
|
22 |
+
.map((f) => ({
|
23 |
+
module_name: `${basename(f, ".spec.ts")}.run`,
|
24 |
+
dir_name: basename(f, ".spec.ts")
|
25 |
+
}));
|
26 |
+
|
27 |
+
export default async function global_setup() {
|
28 |
+
const verbose = process.env.GRADIO_TEST_VERBOSE;
|
29 |
+
|
30 |
+
const port = await find_free_port(7860, 8860);
|
31 |
+
process.env.GRADIO_E2E_TEST_PORT = port;
|
32 |
+
|
33 |
+
process.stdout.write(kl.yellow("\nCreating test gradio app.\n\n"));
|
34 |
+
|
35 |
+
const test_cases = [];
|
36 |
+
// check if there is a testcase file in the same directory as the test file
|
37 |
+
// if there is, append that to the file
|
38 |
+
test_files.forEach((value) => {
|
39 |
+
const test_case_dir = join(ROOT, "demo", value.dir_name);
|
40 |
+
|
41 |
+
readdirSync(test_case_dir)
|
42 |
+
.filter((f) => f.endsWith("_testcase.py"))
|
43 |
+
.forEach((f) => {
|
44 |
+
test_cases.push({
|
45 |
+
module_name: `${value.dir_name}.${basename(f, ".py")}`,
|
46 |
+
dir_name: `${value.dir_name}_${basename(f, ".py")}`
|
47 |
+
});
|
48 |
+
});
|
49 |
+
});
|
50 |
+
|
51 |
+
const all_test_files = test_files.concat(test_cases);
|
52 |
+
const test_app = make_app(all_test_files, port);
|
53 |
+
process.stdout.write(kl.yellow("App created. Starting test server.\n\n"));
|
54 |
+
|
55 |
+
process.stdout.write(kl.bgBlue(" =========================== \n"));
|
56 |
+
process.stdout.write(kl.bgBlue(" === PYTHON STARTUP LOGS === \n"));
|
57 |
+
process.stdout.write(kl.bgBlue(" =========================== \n\n"));
|
58 |
+
|
59 |
+
writeFileSync(TEST_APP_PATH, test_app);
|
60 |
+
|
61 |
+
const app = await spawn_gradio_app(TEST_APP_PATH, port, verbose);
|
62 |
+
|
63 |
+
process.stdout.write(
|
64 |
+
kl.green(`\n\nServer started. Running tests on port ${port}.\n`)
|
65 |
+
);
|
66 |
+
|
67 |
+
return () => {
|
68 |
+
process.stdout.write(kl.green(`\nTests complete, cleaning up!\n`));
|
69 |
+
|
70 |
+
kill_process(app);
|
71 |
+
};
|
72 |
+
}
|
73 |
+
const INFO_RE = /^INFO:/;
|
74 |
+
|
75 |
+
function spawn_gradio_app(app, port, verbose) {
|
76 |
+
const PORT_RE = new RegExp(`:${port}`);
|
77 |
+
|
78 |
+
return new Promise((res, rej) => {
|
79 |
+
const _process = spawn(`python`, [app], {
|
80 |
+
shell: true,
|
81 |
+
stdio: "pipe",
|
82 |
+
cwd: ROOT,
|
83 |
+
env: {
|
84 |
+
...process.env,
|
85 |
+
PYTHONUNBUFFERED: "true",
|
86 |
+
GRADIO_ANALYTICS_ENABLED: "False",
|
87 |
+
GRADIO_IS_E2E_TEST: "1"
|
88 |
+
}
|
89 |
+
});
|
90 |
+
_process.stdout.setEncoding("utf8");
|
91 |
+
|
92 |
+
function std_out(data) {
|
93 |
+
const _data = data.toString();
|
94 |
+
const is_info = INFO_RE.test(_data);
|
95 |
+
|
96 |
+
if (is_info) {
|
97 |
+
process.stdout.write(kl.yellow(_data));
|
98 |
+
}
|
99 |
+
|
100 |
+
if (!is_info) {
|
101 |
+
process.stdout.write(`${_data}\n`);
|
102 |
+
}
|
103 |
+
|
104 |
+
if (PORT_RE.test(_data)) {
|
105 |
+
process.stdout.write(kl.bgBlue("\n =========== END =========== "));
|
106 |
+
|
107 |
+
res(_process);
|
108 |
+
|
109 |
+
if (!verbose) {
|
110 |
+
_process.stdout.off("data", std_out);
|
111 |
+
_process.stderr.off("data", std_out);
|
112 |
+
}
|
113 |
+
}
|
114 |
+
}
|
115 |
+
|
116 |
+
_process.stdout.on("data", std_out);
|
117 |
+
_process.stderr.on("data", std_out);
|
118 |
+
_process.on("exit", () => kill_process(_process));
|
119 |
+
_process.on("close", () => kill_process(_process));
|
120 |
+
_process.on("disconnect", () => kill_process(_process));
|
121 |
+
});
|
122 |
+
}
|
123 |
+
|
124 |
+
function kill_process(process) {
|
125 |
+
process.kill("SIGKILL");
|
126 |
+
}
|
127 |
+
|
128 |
+
function make_app(demos, port) {
|
129 |
+
return `
|
130 |
+
import uvicorn
|
131 |
+
from fastapi import FastAPI
|
132 |
+
import gradio as gr
|
133 |
+
|
134 |
+
${demos.map((obj) => `from demo.${obj.module_name} import demo as ${obj.dir_name}`).join("\n")}
|
135 |
+
|
136 |
+
app = FastAPI()
|
137 |
+
${demos
|
138 |
+
.map(
|
139 |
+
(obj) =>
|
140 |
+
`app = gr.mount_gradio_app(app, ${obj.dir_name}, path="/${obj.dir_name}", max_file_size=${
|
141 |
+
obj.dir_name == "upload_file_limit_test" ? "'15kb'" : "None"
|
142 |
+
})`
|
143 |
+
)
|
144 |
+
.join("\n")}
|
145 |
+
|
146 |
+
config = uvicorn.Config(app, port=${port}, log_level="info")
|
147 |
+
server = uvicorn.Server(config=config)
|
148 |
+
server.run()`;
|
149 |
+
}
|
150 |
+
|
151 |
+
export async function find_free_port(start_port, end_port) {
|
152 |
+
for (let port = start_port; port < end_port; port++) {
|
153 |
+
if (await is_free_port(port)) {
|
154 |
+
return port;
|
155 |
+
}
|
156 |
+
}
|
157 |
+
|
158 |
+
throw new Error(
|
159 |
+
`Could not find free ports: there were not enough ports available.`
|
160 |
+
);
|
161 |
+
}
|
162 |
+
|
163 |
+
export function is_free_port(port) {
|
164 |
+
return new Promise((accept, reject) => {
|
165 |
+
const sock = net.createConnection(port, "127.0.0.1");
|
166 |
+
sock.once("connect", () => {
|
167 |
+
sock.end();
|
168 |
+
accept(false);
|
169 |
+
});
|
170 |
+
sock.once("error", (e) => {
|
171 |
+
sock.destroy();
|
172 |
+
if (e.code === "ECONNREFUSED") {
|
173 |
+
accept(true);
|
174 |
+
} else {
|
175 |
+
reject(e);
|
176 |
+
}
|
177 |
+
});
|
178 |
+
});
|
179 |
+
}
|
.config/playwright.config.js
ADDED
@@ -0,0 +1,73 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { defineConfig, devices } from "@playwright/test";
|
2 |
+
|
3 |
+
const base = defineConfig({
|
4 |
+
use: {
|
5 |
+
screenshot: "only-on-failure",
|
6 |
+
trace: "retain-on-failure",
|
7 |
+
bypassCSP: true,
|
8 |
+
launchOptions: {
|
9 |
+
args: [
|
10 |
+
"--disable-web-security",
|
11 |
+
"--use-fake-device-for-media-stream",
|
12 |
+
"--use-fake-ui-for-media-stream",
|
13 |
+
"--use-file-for-fake-audio-capture=../gradio/test_data/test_audio.wav"
|
14 |
+
]
|
15 |
+
}
|
16 |
+
},
|
17 |
+
expect: { timeout: 10000 },
|
18 |
+
timeout: 10000,
|
19 |
+
testMatch: /.*\.spec\.ts/,
|
20 |
+
testDir: "..",
|
21 |
+
workers: process.env.CI ? 1 : undefined,
|
22 |
+
retries: 3
|
23 |
+
});
|
24 |
+
|
25 |
+
const normal = defineConfig(base, {
|
26 |
+
globalSetup: process.env.CUSTOM_TEST ? undefined : "./playwright-setup.js",
|
27 |
+
projects: [
|
28 |
+
{
|
29 |
+
name: "firefox",
|
30 |
+
use: { ...devices["Desktop Firefox"] },
|
31 |
+
testMatch: /.stream_(audio|video)_out\.spec\.ts/
|
32 |
+
},
|
33 |
+
{
|
34 |
+
name: "chrome",
|
35 |
+
use: {
|
36 |
+
...devices["Desktop Chrome"],
|
37 |
+
permissions: ["clipboard-read", "clipboard-write", "microphone"]
|
38 |
+
},
|
39 |
+
testIgnore: /.stream_(audio|video)_out\.spec\.ts/
|
40 |
+
}
|
41 |
+
]
|
42 |
+
});
|
43 |
+
|
44 |
+
const lite = defineConfig(base, {
|
45 |
+
webServer: {
|
46 |
+
command: "python -m http.server 8000 --directory ../js/lite",
|
47 |
+
url: "http://localhost:8000/",
|
48 |
+
reuseExistingServer: !process.env.CI
|
49 |
+
},
|
50 |
+
testMatch: [
|
51 |
+
"**/file_component_events.spec.ts",
|
52 |
+
"**/kitchen_sink.spec.ts",
|
53 |
+
"**/gallery_component_events.spec.ts",
|
54 |
+
"**/image_remote_url.spec.ts", // To detect the bugs on Lite fixed in https://github.com/gradio-app/gradio/pull/8011 and https://github.com/gradio-app/gradio/pull/8026
|
55 |
+
"**/outbreak_forecast.spec.ts" // To test matplotlib on Lite
|
56 |
+
],
|
57 |
+
workers: 1,
|
58 |
+
retries: 3,
|
59 |
+
timeout: 60000,
|
60 |
+
projects: [
|
61 |
+
{
|
62 |
+
name: "chromium",
|
63 |
+
use: { ...devices["Desktop Chrome"] }
|
64 |
+
},
|
65 |
+
{
|
66 |
+
name: "firefox",
|
67 |
+
use: { ...devices["Desktop Firefox"] },
|
68 |
+
testIgnore: "**/kitchen_sink.*" // This test requires the camera permission but it's not supported on FireFox: https://github.com/microsoft/playwright/issues/11714
|
69 |
+
}
|
70 |
+
]
|
71 |
+
});
|
72 |
+
|
73 |
+
export default !!process.env.GRADIO_E2E_TEST_LITE ? lite : normal;
|
.config/playwright/index.html
ADDED
@@ -0,0 +1,12 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!doctype html>
|
2 |
+
<html lang="en">
|
3 |
+
<head>
|
4 |
+
<meta charset="UTF-8" />
|
5 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
6 |
+
<title>Testing Page</title>
|
7 |
+
</head>
|
8 |
+
<body>
|
9 |
+
<div id="root"></div>
|
10 |
+
<script type="module" src="./index.ts"></script>
|
11 |
+
</body>
|
12 |
+
</html>
|
.config/playwright/index.ts
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
// Import styles, initialize component theme here.
|
2 |
+
// import '../src/common.css';
|
.config/postcss.config.cjs
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
const tailwindcss = require("tailwindcss");
|
2 |
+
const autoprefixer = require("autoprefixer");
|
3 |
+
const nested = require("tailwindcss/nesting");
|
4 |
+
const tw_config = require("./tailwind.config.cjs");
|
5 |
+
|
6 |
+
module.exports = {
|
7 |
+
plugins: [nested, tailwindcss(tw_config), autoprefixer]
|
8 |
+
};
|
.config/setup_vite_tests.ts
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import type { TestingLibraryMatchers } from "@testing-library/jest-dom/matchers";
|
2 |
+
import "@testing-library/jest-dom/vitest";
|
3 |
+
|
4 |
+
declare module "vitest" {
|
5 |
+
interface Assertion<T = any>
|
6 |
+
extends jest.Matchers<void, T>,
|
7 |
+
TestingLibraryMatchers<T, void> {}
|
8 |
+
}
|
.config/svelte.config.js
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import { vitePreprocess } from "@sveltejs/vite-plugin-svelte";
|
2 |
+
|
3 |
+
export default {
|
4 |
+
preprocess: vitePreprocess()
|
5 |
+
};
|
.config/tailwind.config.cjs
ADDED
@@ -0,0 +1,12 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
module.exports = {
|
2 |
+
content: [
|
3 |
+
"./src/**/*.{html,js,svelte,ts}",
|
4 |
+
"**/@gradio/**/*.{html,js,svelte,ts}"
|
5 |
+
],
|
6 |
+
|
7 |
+
theme: {
|
8 |
+
extend: {}
|
9 |
+
},
|
10 |
+
|
11 |
+
plugins: [require("@tailwindcss/forms")]
|
12 |
+
};
|
.config/vitest.config.ts
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
import config from "../js/spa/vite.config";
|
2 |
+
|
3 |
+
export default config;
|
.devcontainer/devcontainer.json
ADDED
@@ -0,0 +1,41 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
// See https://containers.dev
|
2 |
+
{
|
3 |
+
"name": "Python 3",
|
4 |
+
"image": "mcr.microsoft.com/devcontainers/python:1-3.9",
|
5 |
+
|
6 |
+
// See https://containers.dev/features
|
7 |
+
"features": {
|
8 |
+
"ghcr.io/devcontainers/features/git:1": {},
|
9 |
+
"ghcr.io/devcontainers/features/node:1": {},
|
10 |
+
"ghcr.io/devcontainers-contrib/features/ffmpeg-apt-get:1": {}
|
11 |
+
},
|
12 |
+
|
13 |
+
"hostRequirements": {
|
14 |
+
"cpus": 4,
|
15 |
+
"memory": "8gb",
|
16 |
+
"storage": "32gb"
|
17 |
+
},
|
18 |
+
|
19 |
+
"customizations": {
|
20 |
+
"vscode": {
|
21 |
+
"extensions": [
|
22 |
+
"ms-python.python",
|
23 |
+
"ms-python.vscode-pylance",
|
24 |
+
"ms-python.black-formatter",
|
25 |
+
"ms-toolsai.jupyter",
|
26 |
+
"esbenp.prettier-vscode",
|
27 |
+
"svelte.svelte-vscode",
|
28 |
+
"phoenisx.cssvar"
|
29 |
+
],
|
30 |
+
"remote.autoForwardPorts": false
|
31 |
+
}
|
32 |
+
},
|
33 |
+
|
34 |
+
"forwardPorts": [7860, 9876],
|
35 |
+
"portsAttributes": {
|
36 |
+
"7860": { "label": "gradio port" },
|
37 |
+
"9876": { "label": "gradio dev port" }
|
38 |
+
},
|
39 |
+
|
40 |
+
"postCreateCommand": "export NODE_OPTIONS=\"--max-old-space-size=8192\" && chmod +x scripts/install_gradio.sh scripts/install_test_requirements.sh scripts/build_frontend.sh && ./scripts/install_gradio.sh && ./scripts/install_test_requirements.sh && ./scripts/build_frontend.sh"
|
41 |
+
}
|
.dockerignore
ADDED
@@ -0,0 +1,41 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Python build
|
2 |
+
.eggs/
|
3 |
+
gradio.egg-info/*
|
4 |
+
!gradio.egg-info/requires.txt
|
5 |
+
!gradio.egg-info/PKG-INFO
|
6 |
+
dist/
|
7 |
+
*.pyc
|
8 |
+
__pycache__/
|
9 |
+
*.py[cod]
|
10 |
+
*$py.class
|
11 |
+
build/
|
12 |
+
|
13 |
+
# JS build
|
14 |
+
gradio/templates/frontend/static
|
15 |
+
gradio/templates/frontend/cdn
|
16 |
+
|
17 |
+
# Secrets
|
18 |
+
.env
|
19 |
+
|
20 |
+
# Gradio run artifacts
|
21 |
+
*.db
|
22 |
+
*.sqlite3
|
23 |
+
gradio/launches.json
|
24 |
+
gradio/hash_seed.txt
|
25 |
+
|
26 |
+
# Tests
|
27 |
+
.coverage
|
28 |
+
coverage.xml
|
29 |
+
test.txt
|
30 |
+
|
31 |
+
# Demos
|
32 |
+
demo/tmp.zip
|
33 |
+
demo/flagged
|
34 |
+
demo/files/*.avi
|
35 |
+
demo/files/*.mp4
|
36 |
+
|
37 |
+
# Etc
|
38 |
+
.idea/*
|
39 |
+
.DS_Store
|
40 |
+
*.bak
|
41 |
+
workspace.code-workspace
|
.editorconfig
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
root = true
|
3 |
+
|
4 |
+
[{js/**,client/js/**}]
|
5 |
+
end_of_line = lf
|
6 |
+
insert_final_newline = true
|
7 |
+
indent_style = tab
|
8 |
+
tab_width = 2
|
.git-blame-ignore-revs
ADDED
@@ -0,0 +1,14 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# https://github.com/gradio-app/gradio/pull/4487 - refactor components.py to separate files
|
2 |
+
69f36f98535c904e7cac2b4942cecc747ed7443c
|
3 |
+
# Format the codebase
|
4 |
+
cc0cff893f9d7d472788adc2510c123967b384fe
|
5 |
+
# Switch from black to ruff
|
6 |
+
8a70e83db9c7751b46058cdd2514e6bddeef6210
|
7 |
+
# format (#4810)
|
8 |
+
7fa5e766ce0f89f1fb84c329e62c9df9c332120a
|
9 |
+
# lint website
|
10 |
+
4bf301324b3b180fa32166ff1774312b01334c88
|
11 |
+
# format frontend with prettier
|
12 |
+
980b9f60eb49ed81e4957debe7b23a559a4d4b51
|
13 |
+
# Refactor component directories (#5074)
|
14 |
+
1419538ea795caa391e3de809379f10639e9e764
|
.gitattributes
CHANGED
@@ -33,3 +33,38 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
|
33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
36 |
+
demo/blocks_flipper/screenshot.gif filter=lfs diff=lfs merge=lfs -text
|
37 |
+
demo/blocks_neural_instrument_coding/sax.wav filter=lfs diff=lfs merge=lfs -text
|
38 |
+
demo/calculator/screenshot.gif filter=lfs diff=lfs merge=lfs -text
|
39 |
+
demo/chatbot_core_components/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
40 |
+
demo/chatbot_core_components_simple/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
41 |
+
demo/dataset/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
42 |
+
demo/gallery_component/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
43 |
+
demo/hello_world_2/screenshot.gif filter=lfs diff=lfs merge=lfs -text
|
44 |
+
demo/image_mod/screenshot.png filter=lfs diff=lfs merge=lfs -text
|
45 |
+
demo/kitchen_sink/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
46 |
+
demo/rt-detr-object-detection/3285790-hd_1920_1080_30fps.mp4 filter=lfs diff=lfs merge=lfs -text
|
47 |
+
demo/sales_projections/screenshot.gif filter=lfs diff=lfs merge=lfs -text
|
48 |
+
demo/sepia_filter/screenshot.gif filter=lfs diff=lfs merge=lfs -text
|
49 |
+
demo/unispeech-speaker-verification/samples/kirsten_dunst.wav filter=lfs diff=lfs merge=lfs -text
|
50 |
+
demo/video_component/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
51 |
+
guides/assets/hf_demo.mp4 filter=lfs diff=lfs merge=lfs -text
|
52 |
+
guides/cn/assets/hf_demo.mp4 filter=lfs diff=lfs merge=lfs -text
|
53 |
+
js/component-test/src/lib/images/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
54 |
+
js/spa/test/files/file_test.ogg filter=lfs diff=lfs merge=lfs -text
|
55 |
+
js/spa/test/files/world.mp4 filter=lfs diff=lfs merge=lfs -text
|
56 |
+
test/test_files/rotated_image.jpeg filter=lfs diff=lfs merge=lfs -text
|
57 |
+
venv/bin/ruff filter=lfs diff=lfs merge=lfs -text
|
58 |
+
venv/lib/python3.10/site-packages/PIL/.dylibs/libfreetype.6.dylib filter=lfs diff=lfs merge=lfs -text
|
59 |
+
venv/lib/python3.10/site-packages/PIL/.dylibs/libharfbuzz.0.dylib filter=lfs diff=lfs merge=lfs -text
|
60 |
+
venv/lib/python3.10/site-packages/numpy/.dylibs/libgfortran.5.dylib filter=lfs diff=lfs merge=lfs -text
|
61 |
+
venv/lib/python3.10/site-packages/numpy/.dylibs/libscipy_openblas64_.dylib filter=lfs diff=lfs merge=lfs -text
|
62 |
+
venv/lib/python3.10/site-packages/numpy/_core/_multiarray_umath.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
63 |
+
venv/lib/python3.10/site-packages/numpy/_core/_simd.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
64 |
+
venv/lib/python3.10/site-packages/pandas/_libs/algos.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
65 |
+
venv/lib/python3.10/site-packages/pandas/_libs/groupby.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
66 |
+
venv/lib/python3.10/site-packages/pandas/_libs/hashtable.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
67 |
+
venv/lib/python3.10/site-packages/pandas/_libs/interval.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
68 |
+
venv/lib/python3.10/site-packages/pandas/_libs/join.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
69 |
+
venv/lib/python3.10/site-packages/pandas/_libs/tslibs/offsets.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
70 |
+
venv/lib/python3.10/site-packages/pydantic_core/_pydantic_core.cpython-310-darwin.so filter=lfs diff=lfs merge=lfs -text
|
.github/ISSUE_TEMPLATE/bug_report_template.yml
ADDED
@@ -0,0 +1,69 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "\U0001F41E Bug report"
|
2 |
+
description: Report a bug on Gradio
|
3 |
+
labels: ["bug"]
|
4 |
+
body:
|
5 |
+
- type: markdown
|
6 |
+
attributes:
|
7 |
+
value: |
|
8 |
+
Thanks for taking the time to fill out this bug report! Before you get started, please [search to see](https://github.com/gradio-app/gradio/issues) if an issue already exists for the bug you encountered
|
9 |
+
- type: textarea
|
10 |
+
id: bug-description
|
11 |
+
attributes:
|
12 |
+
label: Describe the bug
|
13 |
+
description: Please provide a concise description of what the bug is, in clear English. If you intend to submit a PR for this issue, tell us in the description.
|
14 |
+
placeholder: Bug description
|
15 |
+
validations:
|
16 |
+
required: true
|
17 |
+
- type: checkboxes
|
18 |
+
attributes:
|
19 |
+
label: Have you searched existing issues? 🔎
|
20 |
+
description: Please search to see if an issue already exists for the issue you encountered.
|
21 |
+
options:
|
22 |
+
- label: I have searched and found no existing issues
|
23 |
+
required: true
|
24 |
+
- type: textarea
|
25 |
+
id: reproduction
|
26 |
+
attributes:
|
27 |
+
label: Reproduction
|
28 |
+
description: Please provide a minimal example, with code, that can be run to reproduce the issue. Do NOT provide screenshots of code, or link to external repos or applications. Use ``` to format code blocks.
|
29 |
+
placeholder: Reproduction
|
30 |
+
value: |
|
31 |
+
```python
|
32 |
+
import gradio as gr
|
33 |
+
|
34 |
+
```
|
35 |
+
validations:
|
36 |
+
required: true
|
37 |
+
- type: textarea
|
38 |
+
id: screenshot
|
39 |
+
attributes:
|
40 |
+
label: Screenshot
|
41 |
+
description: If relevant, please include screenshot(s) of your Gradio app so that we can understand what the issue is.
|
42 |
+
- type: textarea
|
43 |
+
id: logs
|
44 |
+
attributes:
|
45 |
+
label: Logs
|
46 |
+
description: "Please include the full stacktrace of the errors you get from Python or Javascript. If you are running in a colab notebooks, you can get the logs with by setting `debug=True`, i.e: `gradio.Interface.launch(debug=True)`"
|
47 |
+
render: shell
|
48 |
+
- type: textarea
|
49 |
+
id: system-info
|
50 |
+
attributes:
|
51 |
+
label: System Info
|
52 |
+
description: Please ensure you are running the latest version of Gradio. You can get the Gradio version and all its dependencies by running `gradio environment`
|
53 |
+
render: shell
|
54 |
+
validations:
|
55 |
+
required: true
|
56 |
+
- type: dropdown
|
57 |
+
id: severity
|
58 |
+
attributes:
|
59 |
+
label: Severity
|
60 |
+
description: Select the severity of this issue
|
61 |
+
options:
|
62 |
+
- I can work around it
|
63 |
+
- Blocking usage of gradio
|
64 |
+
validations:
|
65 |
+
required: true
|
66 |
+
- type: markdown
|
67 |
+
attributes:
|
68 |
+
value: |
|
69 |
+
📌 Please ensure that you have filled all of the required sections above, and that the reproduction you have provided is [minimal, complete, and reproducible](https://stackoverflow.com/help/minimal-reproducible-example). Incomplete issues will be closed.
|
.github/ISSUE_TEMPLATE/config.yml
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
blank_issues_enabled: false
|
2 |
+
contact_links:
|
3 |
+
- name: 💡 General questions
|
4 |
+
url: https://discord.com/invite/feTf9x3ZSB
|
5 |
+
about: Have general questions about how to use Gradio? Please ask in our community Discord for quicker responses
|
.github/ISSUE_TEMPLATE/feature_request.md
ADDED
@@ -0,0 +1,19 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
---
|
2 |
+
name: ⚡ Feature request
|
3 |
+
about: Suggest an improvement or new feature or a new Guide for Gradio
|
4 |
+
title: ''
|
5 |
+
labels: ''
|
6 |
+
assignees: ''
|
7 |
+
|
8 |
+
---
|
9 |
+
- [ ] I have searched to see if a similar issue already exists.
|
10 |
+
|
11 |
+
|
12 |
+
**Is your feature request related to a problem? Please describe.**
|
13 |
+
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
14 |
+
|
15 |
+
**Describe the solution you'd like**
|
16 |
+
A clear and concise description of what you want to happen.
|
17 |
+
|
18 |
+
**Additional context**
|
19 |
+
Add any other context or screenshots about the feature request here.
|
.github/PULL_REQUEST_TEMPLATE.md
ADDED
@@ -0,0 +1,18 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
## Description
|
2 |
+
|
3 |
+
Please include a concise summary, in clear English, of the changes in this pull request. If it closes an issue, please mention it here.
|
4 |
+
|
5 |
+
Closes: #(issue)
|
6 |
+
|
7 |
+
## 🎯 PRs Should Target Issues
|
8 |
+
|
9 |
+
Before your create a PR, please check to see if there is [an existing issue](https://github.com/gradio-app/gradio/issues) for this change. If not, please create an issue before you create this PR, unless the fix is very small.
|
10 |
+
|
11 |
+
Not adhering to this guideline will result in the PR being closed.
|
12 |
+
|
13 |
+
## Tests
|
14 |
+
|
15 |
+
1. PRs will only be merged if tests pass on CI. To run the tests locally, please set up [your Gradio environment locally](https://github.com/gradio-app/gradio/blob/main/CONTRIBUTING.md) and run the tests: `bash scripts/run_all_tests.sh`
|
16 |
+
|
17 |
+
2. You may need to run the linters: `bash scripts/format_backend.sh` and `bash scripts/format_frontend.sh`
|
18 |
+
|
.github/actions/changes/action.yml
ADDED
@@ -0,0 +1,78 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "prepare"
|
2 |
+
description: "Prepare workflow"
|
3 |
+
|
4 |
+
inputs:
|
5 |
+
token:
|
6 |
+
description: "GitHub token"
|
7 |
+
filter:
|
8 |
+
description: "Which filter to use"
|
9 |
+
|
10 |
+
outputs:
|
11 |
+
should_run:
|
12 |
+
description: "Whether to run the workflow"
|
13 |
+
value: ${{ steps.pr.outputs.should_run }}
|
14 |
+
pr_number:
|
15 |
+
description: "PR number"
|
16 |
+
value: ${{ steps.pr.outputs.pr_number }}
|
17 |
+
sha:
|
18 |
+
description: "SHA of the HEAD commit of the PR"
|
19 |
+
value: ${{ steps.pr.outputs.sha }}
|
20 |
+
source_repo:
|
21 |
+
description: "Source repo"
|
22 |
+
value: ${{ steps.pr.outputs.source_repo }}
|
23 |
+
source_branch:
|
24 |
+
description: "Source branch"
|
25 |
+
value: ${{ steps.pr.outputs.source_branch }}
|
26 |
+
labels:
|
27 |
+
description: "Labels on the PR"
|
28 |
+
value: ${{ steps.pr.outputs.labels }}
|
29 |
+
run_id:
|
30 |
+
description: "Run ID"
|
31 |
+
value: ${{ steps.pr.outputs.run_id }}
|
32 |
+
gradio_version:
|
33 |
+
description: "Gradio version"
|
34 |
+
value: ${{ steps.pr.outputs.gradio_version }}
|
35 |
+
|
36 |
+
runs:
|
37 |
+
using: "composite"
|
38 |
+
steps:
|
39 |
+
- uses: actions/checkout@v4
|
40 |
+
- uses: gradio-app/github/actions/filter-paths@main
|
41 |
+
id: changes
|
42 |
+
with:
|
43 |
+
token: ${{ inputs.token }}
|
44 |
+
filter: ${{ inputs.filter }}
|
45 |
+
- name: get gradio version
|
46 |
+
id: gradio_version
|
47 |
+
shell: bash
|
48 |
+
run: |
|
49 |
+
GRADIO_VERSION=$(curl -s https://pypi.org/pypi/gradio/json | grep -o '"version":"[^"]*"' | cut -d'"' -f4 | head -n 1)
|
50 |
+
echo "gradio_version=$GRADIO_VERSION" >> $GITHUB_OUTPUT
|
51 |
+
- name: convert to JSON
|
52 |
+
uses: gradio-app/github/actions/input-to-json@main
|
53 |
+
with:
|
54 |
+
path: output.json
|
55 |
+
sha: ${{ github.event.pull_request.head.sha || github.sha }}
|
56 |
+
source_repo: ${{ github.event.pull_request.head.repo.full_name || github.repository }}
|
57 |
+
source_branch: ${{ github.event.pull_request.head.ref || github.ref }}
|
58 |
+
pr_number: ${{ github.event.pull_request.number || 'false'}}
|
59 |
+
should_run: ${{ steps.changes.outputs.match }}
|
60 |
+
labels: "[${{ join(github.event.pull_request.labels.*.name, ', ') }}]"
|
61 |
+
run_id: ${{ github.run_id }}
|
62 |
+
gradio_version: ${{ steps.gradio_version.outputs.gradio_version }}
|
63 |
+
- name: cat json
|
64 |
+
run: cat output.json
|
65 |
+
shell: bash
|
66 |
+
- name: upload JSON
|
67 |
+
uses: actions/upload-artifact@v4
|
68 |
+
with:
|
69 |
+
name: changes
|
70 |
+
path: output.json
|
71 |
+
- name: set outputs
|
72 |
+
id: pr
|
73 |
+
uses: gradio-app/github/actions/json-to-output@main
|
74 |
+
with:
|
75 |
+
path: output.json
|
76 |
+
- name: echo outputs
|
77 |
+
run: echo "${{ toJson(steps.pr.outputs) }}"
|
78 |
+
shell: bash
|
.github/actions/install-all-deps/action.yml
ADDED
@@ -0,0 +1,92 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "install all deps"
|
2 |
+
description: "Install all deps"
|
3 |
+
|
4 |
+
inputs:
|
5 |
+
skip_build:
|
6 |
+
description: "Skip build"
|
7 |
+
default: "false"
|
8 |
+
build_lite:
|
9 |
+
description: "Build lite"
|
10 |
+
default: "false"
|
11 |
+
test:
|
12 |
+
description: "Test"
|
13 |
+
default: "false"
|
14 |
+
python_version:
|
15 |
+
description: "Python version"
|
16 |
+
default: "3.10"
|
17 |
+
os:
|
18 |
+
description: "OS"
|
19 |
+
default: "ubuntu-latest"
|
20 |
+
outputs:
|
21 |
+
venv_activate:
|
22 |
+
description: "Venv activate"
|
23 |
+
value: ${{ steps.venv.outputs.venv_activate }}
|
24 |
+
|
25 |
+
runs:
|
26 |
+
using: "composite"
|
27 |
+
steps:
|
28 |
+
- name: Set venv binary path (linux)
|
29 |
+
if: ${{ inputs.os == 'ubuntu-latest' }}
|
30 |
+
shell: bash
|
31 |
+
run: |
|
32 |
+
echo "VENV_ACTIVATE=venv/bin/activate" >> $GITHUB_ENV
|
33 |
+
- name: Set venv binary path (windows)
|
34 |
+
if: ${{ inputs.os == 'windows-latest' }}
|
35 |
+
shell: bash
|
36 |
+
run: |
|
37 |
+
echo "VENV_ACTIVATE=venv\\\Scripts\\\activate" >> $GITHUB_ENV
|
38 |
+
- name: log venv binary path
|
39 |
+
id: venv
|
40 |
+
shell: bash
|
41 |
+
run: |
|
42 |
+
echo "venv_activate=$VENV_ACTIVATE" >> $GITHUB_OUTPUT
|
43 |
+
- name: Install Python
|
44 |
+
uses: actions/setup-python@v5
|
45 |
+
with:
|
46 |
+
python-version: ${{ inputs.python_version }}
|
47 |
+
cache: pip
|
48 |
+
cache-dependency-path: |
|
49 |
+
client/python/requirements.txt
|
50 |
+
requirements.txt
|
51 |
+
test/requirements.txt
|
52 |
+
- name: Create env
|
53 |
+
shell: bash
|
54 |
+
run: |
|
55 |
+
python -m pip install --upgrade virtualenv
|
56 |
+
python -m virtualenv venv
|
57 |
+
# - uses: actions/cache@v4
|
58 |
+
# id: cache
|
59 |
+
# with:
|
60 |
+
# path: |
|
61 |
+
# venv/*
|
62 |
+
# client/python/venv
|
63 |
+
# restore-keys: |
|
64 |
+
# gradio-lib-${{inputs.python_version}}-${{inputs.os}}-latest-pip-
|
65 |
+
# key: "gradio-lib-${{inputs.python_version}}-${{inputs.os}}-latest-pip-${{ hashFiles('client/python/requirements.txt') }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('test/requirements.txt') }}-${{ hashFiles('client/python/test/requirements.txt') }}${{ inputs.test == 'true' && '-test' || ''}}"
|
66 |
+
- name: Install ffmpeg
|
67 |
+
uses: FedericoCarboni/setup-ffmpeg@583042d32dd1cabb8bd09df03bde06080da5c87c # @v2
|
68 |
+
- name: Install test dependencies
|
69 |
+
if: inputs.test == 'true'
|
70 |
+
# && steps.cache.outputs.cache-hit != 'true'
|
71 |
+
shell: bash
|
72 |
+
run: |
|
73 |
+
. ${{ env.VENV_ACTIVATE }}
|
74 |
+
python -m pip install -r test/requirements.txt
|
75 |
+
python -m pip install -r client/python/test/requirements.txt
|
76 |
+
- name: Install Gradio and Client Libraries Locally (Linux)
|
77 |
+
shell: bash
|
78 |
+
run: |
|
79 |
+
. ${{ env.VENV_ACTIVATE }}
|
80 |
+
python -m pip install -e client/python
|
81 |
+
python -m pip install -e ".[oauth]"
|
82 |
+
- name: install-frontend
|
83 |
+
uses: "gradio-app/gradio/.github/actions/install-frontend-deps@main"
|
84 |
+
with:
|
85 |
+
skip_build: ${{ inputs.skip_build }}
|
86 |
+
build_lite: ${{ inputs.build_lite }}
|
87 |
+
- name: generate json
|
88 |
+
shell: bash
|
89 |
+
if: inputs.os == 'ubuntu-latest'
|
90 |
+
run: |
|
91 |
+
. ${{ env.VENV_ACTIVATE }}
|
92 |
+
pip install boto3 && python js/_website/generate_jsons/generate.py
|
.github/actions/install-frontend-deps/action.yml
ADDED
@@ -0,0 +1,51 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "install frontend"
|
2 |
+
description: "Install frontend deps"
|
3 |
+
|
4 |
+
inputs:
|
5 |
+
skip_build:
|
6 |
+
description: "Skip build"
|
7 |
+
default: "false"
|
8 |
+
build_lite:
|
9 |
+
description: "Build lite"
|
10 |
+
default: "false"
|
11 |
+
|
12 |
+
runs:
|
13 |
+
using: "composite"
|
14 |
+
steps:
|
15 |
+
# - uses: actions/cache@v4
|
16 |
+
# id: frontend-cache
|
17 |
+
# with:
|
18 |
+
# path: |
|
19 |
+
# gradio/templates/*
|
20 |
+
# key: gradio-lib-front-end-${{ hashFiles('js/**', 'client/js/**')}}
|
21 |
+
- name: Install pnpm
|
22 |
+
uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # @v4
|
23 |
+
with:
|
24 |
+
version: 9.1.x
|
25 |
+
- uses: actions/setup-node@v4
|
26 |
+
with:
|
27 |
+
node-version: 18
|
28 |
+
cache: pnpm
|
29 |
+
cache-dependency-path: pnpm-lock.yaml
|
30 |
+
- name: Install deps
|
31 |
+
shell: bash
|
32 |
+
run: pnpm i --frozen-lockfile --ignore-scripts
|
33 |
+
- name: Build Css
|
34 |
+
shell: bash
|
35 |
+
run: pnpm css
|
36 |
+
- name: Build frontend
|
37 |
+
if: inputs.skip_build == 'false'
|
38 |
+
# && steps.frontend-cache.outputs.cache-hit != 'true'
|
39 |
+
shell: bash
|
40 |
+
run: pnpm build
|
41 |
+
- name: generate types
|
42 |
+
if: inputs.skip_build == 'false' || inputs.build_lite == 'true'
|
43 |
+
shell: bash
|
44 |
+
run: pnpm package
|
45 |
+
- name: Build frontend lite
|
46 |
+
if: inputs.build_lite == 'true'
|
47 |
+
shell: bash
|
48 |
+
run: |
|
49 |
+
. venv/bin/activate
|
50 |
+
python -m pip install -U build hatch packaging>=23.2 # packaging>=23.2 is needed to build Lite due to https://github.com/pypa/hatch/issues/1381
|
51 |
+
pnpm build:lite
|
.github/configs/semgrep_rules.yaml
ADDED
@@ -0,0 +1,110 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
rules:
|
2 |
+
- id: third-party-action-not-pinned-to-commit-sha
|
3 |
+
patterns:
|
4 |
+
- pattern-inside: "{steps: ...}"
|
5 |
+
- pattern: |
|
6 |
+
uses: "$USES"
|
7 |
+
- metavariable-pattern:
|
8 |
+
metavariable: $USES
|
9 |
+
language: generic
|
10 |
+
patterns:
|
11 |
+
- pattern-not-regex: ^[.]/
|
12 |
+
- pattern-not-regex: ^actions/
|
13 |
+
- pattern-not-regex: ^github/
|
14 |
+
- pattern-not-regex: ^gradio-app/gradio
|
15 |
+
- pattern-not-regex: ^gradio-app/github
|
16 |
+
- pattern-not-regex: "@[0-9a-f]{40}$"
|
17 |
+
- pattern-not-regex: ^docker://.*@sha256:[0-9a-f]{64}$
|
18 |
+
- pattern-not-regex: ^docker://docker$
|
19 |
+
message:
|
20 |
+
An action sourced from a third-party repository on GitHub is not pinned
|
21 |
+
to a full length commit SHA. Pinning an action to a full length commit SHA
|
22 |
+
is currently the only way to use an action as an immutable release.
|
23 |
+
Pinning to a particular SHA helps mitigate the risk of a bad actor adding
|
24 |
+
a backdoor to the action's repository, as they would need to generate a
|
25 |
+
SHA-1 collision for a valid Git object payload.
|
26 |
+
languages:
|
27 |
+
- yaml
|
28 |
+
severity: WARNING
|
29 |
+
metadata:
|
30 |
+
cwe:
|
31 |
+
- "CWE-1357: Reliance on Insufficiently Trustworthy Component"
|
32 |
+
- "CWE-353: Missing Support for Integrity Check"
|
33 |
+
owasp: A06:2021 - Vulnerable and Outdated Components
|
34 |
+
references:
|
35 |
+
- https://owasp.org/Top10/A06_2021-Vulnerable_and_Outdated_Components
|
36 |
+
- https://docs.github.com/en/actions/security-guides/security-hardening-for-github-actions#using-third-party-actions
|
37 |
+
category: security
|
38 |
+
technology:
|
39 |
+
- github-actions
|
40 |
+
subcategory:
|
41 |
+
- vuln
|
42 |
+
likelihood: LOW
|
43 |
+
impact: LOW
|
44 |
+
confidence: HIGH
|
45 |
+
license: Commons Clause License Condition v1.0[LGPL-2.1-only]
|
46 |
+
vulnerability_class:
|
47 |
+
- Cryptographic Issues
|
48 |
+
- Other
|
49 |
+
|
50 |
+
- id: insecure-file-permissions
|
51 |
+
languages:
|
52 |
+
- python
|
53 |
+
severity: ERROR
|
54 |
+
message: These permissions `$BITS` are widely permissive and grant access to
|
55 |
+
more people than may be necessary. A good default is `0o644` which gives
|
56 |
+
read and write access to yourself and read access to everyone else.
|
57 |
+
patterns:
|
58 |
+
- pattern-inside: os.$METHOD(...)
|
59 |
+
- pattern-either:
|
60 |
+
- patterns:
|
61 |
+
- pattern: os.$METHOD($FILE, $BITS, ...)
|
62 |
+
- metavariable-comparison:
|
63 |
+
comparison: $BITS >= 0o650 and $BITS < 0o100000
|
64 |
+
- patterns:
|
65 |
+
- pattern: os.$METHOD($FILE, $BITS)
|
66 |
+
- metavariable-comparison:
|
67 |
+
comparison: $BITS >= 0o100650
|
68 |
+
- patterns:
|
69 |
+
- pattern: os.$METHOD($FILE, $BITS, ...)
|
70 |
+
- metavariable-pattern:
|
71 |
+
metavariable: $BITS
|
72 |
+
patterns:
|
73 |
+
- pattern-either:
|
74 |
+
- pattern: <... stat.S_IWGRP ...>
|
75 |
+
- pattern: <... stat.S_IXGRP ...>
|
76 |
+
- pattern: <... stat.S_IWOTH ...>
|
77 |
+
- pattern: <... stat.S_IXOTH ...>
|
78 |
+
- pattern: <... stat.S_IRWXO ...>
|
79 |
+
- pattern: <... stat.S_IRWXG ...>
|
80 |
+
- patterns:
|
81 |
+
- pattern: os.$METHOD($FILE, $EXPR | $MOD, ...)
|
82 |
+
- metavariable-comparison:
|
83 |
+
comparison: $MOD == 0o111
|
84 |
+
- metavariable-pattern:
|
85 |
+
metavariable: $METHOD
|
86 |
+
patterns:
|
87 |
+
- pattern-either:
|
88 |
+
- pattern: chmod
|
89 |
+
- pattern: lchmod
|
90 |
+
- pattern: fchmod
|
91 |
+
metadata:
|
92 |
+
category: security
|
93 |
+
owasp:
|
94 |
+
- A01:2021 - Broken Access Control
|
95 |
+
cwe:
|
96 |
+
- "CWE-276: Incorrect Default Permissions"
|
97 |
+
technology:
|
98 |
+
- python
|
99 |
+
references:
|
100 |
+
- https://owasp.org/Top10/A01_2021-Broken_Access_Control
|
101 |
+
cwe2022-top25: true
|
102 |
+
cwe2021-top25: true
|
103 |
+
subcategory:
|
104 |
+
- vuln
|
105 |
+
likelihood: LOW
|
106 |
+
impact: MEDIUM
|
107 |
+
confidence: MEDIUM
|
108 |
+
license: Commons Clause License Condition v1.0[LGPL-2.1-only]
|
109 |
+
vulnerability_class:
|
110 |
+
- Improper Authorization
|
.github/filters.json
ADDED
@@ -0,0 +1,51 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"gradio": [
|
3 |
+
"client/python/**",
|
4 |
+
"gradio/**",
|
5 |
+
"requirements.txt",
|
6 |
+
".github/**",
|
7 |
+
"scripts/**",
|
8 |
+
"test/**"
|
9 |
+
],
|
10 |
+
"js": [
|
11 |
+
"js/**",
|
12 |
+
"client/js/**",
|
13 |
+
".github/**",
|
14 |
+
"package.json",
|
15 |
+
"pnpm-lock.yaml",
|
16 |
+
"tsconfig.json",
|
17 |
+
".config/**"
|
18 |
+
],
|
19 |
+
"functional": [
|
20 |
+
".github/**",
|
21 |
+
"client/**",
|
22 |
+
"demo/**",
|
23 |
+
"gradio/**",
|
24 |
+
"js/**",
|
25 |
+
"scripts/**",
|
26 |
+
|
27 |
+
"globals.d.ts",
|
28 |
+
"package.json",
|
29 |
+
"pnpm-lock.yaml",
|
30 |
+
"pyproject.toml",
|
31 |
+
"requirements.txt",
|
32 |
+
".config/**"
|
33 |
+
],
|
34 |
+
"visual": [
|
35 |
+
".github/workflows/deploy-chromatic.yml",
|
36 |
+
"js!(/_website)/**",
|
37 |
+
"package.json"
|
38 |
+
],
|
39 |
+
"website": [
|
40 |
+
"js/_website/**",
|
41 |
+
"package.json",
|
42 |
+
"pnpm-lock.yaml",
|
43 |
+
"guides/**",
|
44 |
+
"README.md",
|
45 |
+
"CHANGELOG.md",
|
46 |
+
"gradio/**",
|
47 |
+
"client/**",
|
48 |
+
"demo/**",
|
49 |
+
".github/deploy-website.yml"
|
50 |
+
]
|
51 |
+
}
|
.github/stale
ADDED
@@ -0,0 +1,17 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Number of days of inactivity before an issue becomes stale
|
2 |
+
daysUntilStale: 30
|
3 |
+
# Number of days of inactivity before a stale issue is closed
|
4 |
+
daysUntilClose: 7
|
5 |
+
# Issues with these labels will never be considered stale
|
6 |
+
exemptLabels:
|
7 |
+
- pinned
|
8 |
+
- security
|
9 |
+
# Label to use when marking an issue as stale
|
10 |
+
staleLabel: wontfix
|
11 |
+
# Comment to post when marking an issue as stale. Set to `false` to disable
|
12 |
+
markComment: >
|
13 |
+
This issue has been automatically marked as stale because it has not had
|
14 |
+
recent activity. It will be closed if no further activity occurs. Thank you
|
15 |
+
for your contributions.
|
16 |
+
# Comment to post when closing a stale issue. Set to `false` to disable
|
17 |
+
closeComment: false
|
.github/workflows/comment-queue.yml
ADDED
@@ -0,0 +1,37 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: Comment on pull request without race conditions
|
2 |
+
|
3 |
+
on:
|
4 |
+
workflow_call:
|
5 |
+
inputs:
|
6 |
+
pr_number:
|
7 |
+
type: string
|
8 |
+
message:
|
9 |
+
required: true
|
10 |
+
type: string
|
11 |
+
tag:
|
12 |
+
required: false
|
13 |
+
type: string
|
14 |
+
default: "previews"
|
15 |
+
additional_text:
|
16 |
+
required: false
|
17 |
+
type: string
|
18 |
+
default: ""
|
19 |
+
secrets:
|
20 |
+
gh_token:
|
21 |
+
required: true
|
22 |
+
|
23 |
+
jobs:
|
24 |
+
comment:
|
25 |
+
environment: comment_pr
|
26 |
+
concurrency:
|
27 |
+
group: ${{inputs.pr_number || inputs.tag}}
|
28 |
+
runs-on: ubuntu-latest
|
29 |
+
steps:
|
30 |
+
- name: comment on pr
|
31 |
+
uses: "gradio-app/github/actions/comment-pr@main"
|
32 |
+
with:
|
33 |
+
gh_token: ${{ secrets.gh_token }}
|
34 |
+
tag: ${{ inputs.tag }}
|
35 |
+
pr_number: ${{ inputs.pr_number}}
|
36 |
+
message: ${{ inputs.message }}
|
37 |
+
additional_text: ${{ inputs.additional_text }}
|
.github/workflows/delete-stale-spaces.yml
ADDED
@@ -0,0 +1,42 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# safe runs from main
|
2 |
+
|
3 |
+
name: Delete Stale Spaces
|
4 |
+
|
5 |
+
on:
|
6 |
+
schedule:
|
7 |
+
- cron: "0 0 * * *"
|
8 |
+
workflow_dispatch:
|
9 |
+
inputs:
|
10 |
+
daysStale:
|
11 |
+
description: "How stale a space needs to be to be deleted (days)"
|
12 |
+
required: true
|
13 |
+
default: "7"
|
14 |
+
|
15 |
+
permissions: {}
|
16 |
+
|
17 |
+
jobs:
|
18 |
+
delete-old-spaces:
|
19 |
+
permissions:
|
20 |
+
contents: read
|
21 |
+
environment: deploy_spaces
|
22 |
+
runs-on: ubuntu-latest
|
23 |
+
steps:
|
24 |
+
- uses: actions/checkout@v4
|
25 |
+
- name: Install Python
|
26 |
+
uses: actions/setup-python@v5
|
27 |
+
with:
|
28 |
+
python-version: "3.10"
|
29 |
+
- name: Install pip
|
30 |
+
run: python -m pip install pip wheel requests
|
31 |
+
- name: Install Hub Client Library
|
32 |
+
run: pip install huggingface-hub==0.9.1
|
33 |
+
- name: Set daysStale
|
34 |
+
env:
|
35 |
+
DEFAULT_DAYS_STALE: "7"
|
36 |
+
DAYS_STALE: ${{ github.event.inputs.daysStale || env.DEFAULT_DAYS_STALE}}
|
37 |
+
run: echo DAYS_STALE= "$DAYS_STALE" >> $GITHUB_ENV
|
38 |
+
- name: Find and delete stale spaces
|
39 |
+
run: |
|
40 |
+
python scripts/delete_old_spaces.py $DAYS_STALE \
|
41 |
+
gradio-pr-deploys \
|
42 |
+
${{ secrets.SPACES_DEPLOY_TOKEN }}
|
.github/workflows/generate-changeset.yml
ADDED
@@ -0,0 +1,96 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: Generate changeset
|
2 |
+
on:
|
3 |
+
workflow_run:
|
4 |
+
workflows: ["trigger-changeset"]
|
5 |
+
types:
|
6 |
+
- completed
|
7 |
+
|
8 |
+
env:
|
9 |
+
CI: true
|
10 |
+
NODE_OPTIONS: "--max-old-space-size=4096"
|
11 |
+
|
12 |
+
concurrency:
|
13 |
+
group: "${{ github.event.workflow_run.head_repository.full_name }}-${{ github.event.workflow_run.head_branch }}-${{ github.workflow_ref }}"
|
14 |
+
cancel-in-progress: true
|
15 |
+
|
16 |
+
permissions: {}
|
17 |
+
|
18 |
+
jobs:
|
19 |
+
get-pr:
|
20 |
+
runs-on: ubuntu-latest
|
21 |
+
permissions:
|
22 |
+
contents: read
|
23 |
+
pull-requests: read
|
24 |
+
if: github.event.workflow_run.conclusion == 'success'
|
25 |
+
outputs:
|
26 |
+
found_pr: ${{ steps.pr_details.outputs.found_pr }}
|
27 |
+
pr_number: ${{ steps.pr_details.outputs.pr_number }}
|
28 |
+
source_repo: ${{ steps.pr_details.outputs.source_repo }}
|
29 |
+
source_branch: ${{ steps.pr_details.outputs.source_branch }}
|
30 |
+
steps:
|
31 |
+
- name: get pr details
|
32 |
+
id: pr_details
|
33 |
+
uses: gradio-app/github/actions/find-pr@main
|
34 |
+
with:
|
35 |
+
github_token: ${{ secrets.GITHUB_TOKEN }}
|
36 |
+
comment-changes-start:
|
37 |
+
uses: "./.github/workflows/comment-queue.yml"
|
38 |
+
needs: get-pr
|
39 |
+
secrets:
|
40 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
41 |
+
with:
|
42 |
+
pr_number: ${{ needs.get-pr.outputs.pr_number }}
|
43 |
+
message: changes~pending~null
|
44 |
+
version:
|
45 |
+
permissions:
|
46 |
+
contents: read
|
47 |
+
environment: changeset
|
48 |
+
name: version
|
49 |
+
needs: get-pr
|
50 |
+
runs-on: ubuntu-latest
|
51 |
+
if: needs.get-pr.outputs.found_pr == 'true'
|
52 |
+
outputs:
|
53 |
+
skipped: ${{ steps.version.outputs.skipped }}
|
54 |
+
comment_url: ${{ steps.version.outputs.comment_url }}
|
55 |
+
steps:
|
56 |
+
- uses: actions/checkout@v4
|
57 |
+
with:
|
58 |
+
repository: ${{ needs.get-pr.outputs.source_repo }}
|
59 |
+
ref: ${{ needs.get-pr.outputs.source_branch }}
|
60 |
+
fetch-depth: 0
|
61 |
+
token: ${{ secrets.COMMENT_TOKEN }}
|
62 |
+
- name: generate changeset
|
63 |
+
id: version
|
64 |
+
uses: "gradio-app/github/actions/generate-changeset@main"
|
65 |
+
with:
|
66 |
+
github_token: ${{ secrets.CHANGESET_GITHUB_TOKEN }}
|
67 |
+
main_pkg: gradio
|
68 |
+
pr_number: ${{ needs.get-pr.outputs.pr_number }}
|
69 |
+
branch_name: ${{ needs.get-pr.outputs.source_branch }}
|
70 |
+
comment-changes-skipped:
|
71 |
+
uses: "./.github/workflows/comment-queue.yml"
|
72 |
+
needs: [get-pr, version]
|
73 |
+
if: needs.version.result == 'success' && needs.version.outputs.skipped == 'true'
|
74 |
+
secrets:
|
75 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
76 |
+
with:
|
77 |
+
pr_number: ${{ needs.get-pr.outputs.pr_number }}
|
78 |
+
message: changes~warning~https://github.com/gradio-app/gradio/actions/runs/${{github.run_id}}/
|
79 |
+
comment-changes-success:
|
80 |
+
uses: "./.github/workflows/comment-queue.yml"
|
81 |
+
needs: [get-pr, version]
|
82 |
+
if: needs.version.result == 'success' && needs.version.outputs.skipped == 'false'
|
83 |
+
secrets:
|
84 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
85 |
+
with:
|
86 |
+
pr_number: ${{ needs.get-pr.outputs.pr_number }}
|
87 |
+
message: changes~success~${{ needs.version.outputs.comment_url }}
|
88 |
+
comment-changes-failure:
|
89 |
+
uses: "./.github/workflows/comment-queue.yml"
|
90 |
+
needs: [get-pr, version]
|
91 |
+
if: always() && needs.version.result == 'failure'
|
92 |
+
secrets:
|
93 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
94 |
+
with:
|
95 |
+
pr_number: ${{ needs.get-pr.outputs.pr_number }}
|
96 |
+
message: changes~failure~https://github.com/gradio-app/gradio/actions/runs/${{github.run_id}}/
|
.github/workflows/npm-previews.yml
ADDED
@@ -0,0 +1,53 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "npm"
|
2 |
+
|
3 |
+
on:
|
4 |
+
pull_request:
|
5 |
+
|
6 |
+
env:
|
7 |
+
CI: true
|
8 |
+
PLAYWRIGHT_SKIP_BROWSER_DOWNLOAD: "1"
|
9 |
+
NODE_OPTIONS: "--max-old-space-size=4096"
|
10 |
+
|
11 |
+
concurrency:
|
12 |
+
group: "${{ github.event.pull_request.number }}-${{ github.ref_name }}-${{ github.workflow }}"
|
13 |
+
cancel-in-progress: true
|
14 |
+
|
15 |
+
permissions: {}
|
16 |
+
|
17 |
+
jobs:
|
18 |
+
changes:
|
19 |
+
permissions:
|
20 |
+
contents: read
|
21 |
+
pull-requests: read
|
22 |
+
name: "changes"
|
23 |
+
runs-on: ubuntu-latest
|
24 |
+
outputs:
|
25 |
+
should_run: ${{ steps.changes.outputs.should_run }}
|
26 |
+
sha: ${{ steps.changes.outputs.sha }}
|
27 |
+
pr_number: ${{ steps.changes.outputs.pr_number }}
|
28 |
+
source_branch: ${{ steps.changes.outputs.source_branch }}
|
29 |
+
source_repo: ${{ steps.changes.outputs.source_repo }}
|
30 |
+
steps:
|
31 |
+
- uses: actions/checkout@v4
|
32 |
+
- uses: "gradio-app/gradio/.github/actions/changes@main"
|
33 |
+
id: changes
|
34 |
+
with:
|
35 |
+
filter: "js"
|
36 |
+
token: ${{ secrets.GITHUB_TOKEN }}
|
37 |
+
preview:
|
38 |
+
permissions:
|
39 |
+
contents: read
|
40 |
+
name: npm-previews
|
41 |
+
runs-on: ubuntu-22.04
|
42 |
+
needs: changes
|
43 |
+
if: needs.changes.outputs.should_run == 'true'
|
44 |
+
steps:
|
45 |
+
- uses: actions/checkout@v4
|
46 |
+
- name: install dependencies
|
47 |
+
uses: "gradio-app/gradio/.github/actions/install-frontend-deps@main"
|
48 |
+
with:
|
49 |
+
skip_build: true
|
50 |
+
- name: build client
|
51 |
+
run: pnpm --filter @gradio/client --filter @gradio/wasm --filter @gradio/preview build
|
52 |
+
- name: publish npm previews
|
53 |
+
run: pnpx pkg-pr-new publish './js/*' './client/js' --comment=off
|
.github/workflows/previews-build.yml
ADDED
@@ -0,0 +1,90 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "previews-build"
|
2 |
+
|
3 |
+
on:
|
4 |
+
workflow_dispatch:
|
5 |
+
pull_request:
|
6 |
+
|
7 |
+
env:
|
8 |
+
CI: true
|
9 |
+
PLAYWRIGHT_SKIP_BROWSER_DOWNLOAD: "1"
|
10 |
+
NODE_OPTIONS: "--max-old-space-size=4096"
|
11 |
+
|
12 |
+
concurrency:
|
13 |
+
group: "${{ github.event.pull_request.number }}-${{ github.ref_name }}-${{ github.workflow }}"
|
14 |
+
cancel-in-progress: true
|
15 |
+
|
16 |
+
permissions: {}
|
17 |
+
|
18 |
+
jobs:
|
19 |
+
changes:
|
20 |
+
name: "changes"
|
21 |
+
runs-on: ubuntu-latest
|
22 |
+
permissions:
|
23 |
+
contents: read
|
24 |
+
pull-requests: read
|
25 |
+
outputs:
|
26 |
+
should_run: ${{ steps.changes.outputs.should_run }}
|
27 |
+
sha: ${{ steps.changes.outputs.sha }}
|
28 |
+
gradio_version: ${{ steps.changes.outputs.gradio_version }}
|
29 |
+
steps:
|
30 |
+
- uses: actions/checkout@v4
|
31 |
+
- uses: "gradio-app/gradio/.github/actions/changes@main"
|
32 |
+
id: changes
|
33 |
+
with:
|
34 |
+
filter: "functional"
|
35 |
+
token: ${{ secrets.GITHUB_TOKEN }}
|
36 |
+
build:
|
37 |
+
permissions:
|
38 |
+
contents: read
|
39 |
+
name: "previews-build"
|
40 |
+
runs-on: ubuntu-22.04
|
41 |
+
needs: changes
|
42 |
+
if: needs.changes.outputs.should_run == 'true' || github.event_name == 'workflow_dispatch'
|
43 |
+
steps:
|
44 |
+
- uses: actions/checkout@v4
|
45 |
+
- name: install dependencies
|
46 |
+
uses: "gradio-app/gradio/.github/actions/install-all-deps@main"
|
47 |
+
with:
|
48 |
+
python_version: "3.10"
|
49 |
+
build_lite: "true"
|
50 |
+
- name: Package Lite NPM package
|
51 |
+
working-directory: js/lite
|
52 |
+
run: pnpm pack --pack-destination .
|
53 |
+
- name: Upload Lite NPM package
|
54 |
+
uses: actions/upload-artifact@v4
|
55 |
+
with:
|
56 |
+
name: gradio-lite-tar
|
57 |
+
path: js/lite/gradio-lite-*.tgz
|
58 |
+
- name: install deps
|
59 |
+
run: python -m pip install build
|
60 |
+
- name: Build pr package
|
61 |
+
run: |
|
62 |
+
python -c 'import json; j = json.load(open("gradio/package.json")); j["version"] = "${{ needs.changes.outputs.gradio_version }}"; json.dump(j, open("gradio/package.json", "w"))'
|
63 |
+
python3 -m build -w
|
64 |
+
- name: Upload Python package
|
65 |
+
uses: actions/upload-artifact@v4
|
66 |
+
with:
|
67 |
+
name: gradio-build
|
68 |
+
path: dist/gradio-${{ needs.changes.outputs.gradio_version }}-py3-none-any.whl
|
69 |
+
- name: copy demos
|
70 |
+
uses: "gradio-app/github/actions/copy-demos@main"
|
71 |
+
with:
|
72 |
+
gradio_version: "https://gradio-pypi-previews.s3.amazonaws.com/${{ needs.changes.outputs.sha }}/gradio-${{ needs.changes.outputs.gradio_version }}-py3-none-any.whl"
|
73 |
+
gradio_client_version: "gradio-client @ git+https://github.com/gradio-app/gradio@${{ needs.changes.outputs.sha }}#subdirectory=client/python"
|
74 |
+
- name: upload demos
|
75 |
+
uses: actions/upload-artifact@v4
|
76 |
+
with:
|
77 |
+
name: all_demos
|
78 |
+
path: demo
|
79 |
+
- name: Create JS client tarball
|
80 |
+
id: create_js_tarball
|
81 |
+
continue-on-error: true
|
82 |
+
run: |
|
83 |
+
cd client/js
|
84 |
+
tarball_name=$(npm pack)
|
85 |
+
echo "tarball_name=client/js/$tarball_name" >> $GITHUB_OUTPUT
|
86 |
+
- name: Upload JS client tarball artifact
|
87 |
+
uses: actions/upload-artifact@v4
|
88 |
+
with:
|
89 |
+
name: js-client-tarball
|
90 |
+
path: ${{ steps.create_js_tarball.outputs.tarball_name }}
|
.github/workflows/previews-deploy.yml
ADDED
@@ -0,0 +1,176 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: "previews-deploy"
|
2 |
+
|
3 |
+
on:
|
4 |
+
workflow_run:
|
5 |
+
workflows: ["previews-build"]
|
6 |
+
types:
|
7 |
+
- completed
|
8 |
+
|
9 |
+
concurrency:
|
10 |
+
group: "${{ github.event.workflow_run.head_repository.full_name }}-${{ github.event.workflow_run.head_branch }}-${{ github.workflow_ref }}"
|
11 |
+
cancel-in-progress: true
|
12 |
+
|
13 |
+
jobs:
|
14 |
+
changes:
|
15 |
+
name: "changes"
|
16 |
+
runs-on: ubuntu-latest
|
17 |
+
if: github.event.workflow_run.conclusion == 'success'
|
18 |
+
permissions:
|
19 |
+
actions: read
|
20 |
+
outputs:
|
21 |
+
should_run: ${{ steps.json.outputs.should_run }}
|
22 |
+
sha: ${{ steps.json.outputs.sha }}
|
23 |
+
pr_number: ${{ steps.json.outputs.pr_number }}
|
24 |
+
source_branch: ${{ steps.json.outputs.source_branch }}
|
25 |
+
source_repo: ${{ steps.json.outputs.source_repo }}
|
26 |
+
labels: ${{ steps.json.outputs.labels }}
|
27 |
+
run_id: ${{ steps.json.outputs.run_id }}
|
28 |
+
gradio_version: ${{ steps.json.outputs.gradio_version }}
|
29 |
+
steps:
|
30 |
+
- name: Download artifact
|
31 |
+
uses: actions/download-artifact@v4
|
32 |
+
with:
|
33 |
+
name: changes
|
34 |
+
github-token: ${{ secrets.GITHUB_TOKEN }}
|
35 |
+
run-id: ${{ github.event.workflow_run.id }}
|
36 |
+
- uses: gradio-app/github/actions/json-to-output@main
|
37 |
+
id: json
|
38 |
+
with:
|
39 |
+
path: output.json
|
40 |
+
|
41 |
+
comment-spaces-start:
|
42 |
+
needs: changes
|
43 |
+
uses: "./.github/workflows/comment-queue.yml"
|
44 |
+
if: ${{ needs.changes.outputs.should_run == 'true' }}
|
45 |
+
secrets:
|
46 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
47 |
+
with:
|
48 |
+
pr_number: ${{ needs.changes.outputs.pr_number }}
|
49 |
+
message: spaces~pending~null
|
50 |
+
deploy:
|
51 |
+
name: "previews-deploy"
|
52 |
+
environment: deploy_spaces
|
53 |
+
outputs:
|
54 |
+
space_url: ${{ steps.upload-demo.outputs.SPACE_URL }}
|
55 |
+
js_tarball_url: ${{ steps.upload_js_tarball.outputs.js_tarball_url }}
|
56 |
+
needs: changes
|
57 |
+
if: ${{ (github.event.workflow_run.conclusion == 'success' && needs.changes.outputs.should_run == 'true') || github.event.workflow_run.event == 'workflow_dispatch' }}
|
58 |
+
runs-on: ubuntu-latest
|
59 |
+
permissions:
|
60 |
+
actions: read
|
61 |
+
steps:
|
62 |
+
- name: Download all artifacts
|
63 |
+
uses: actions/download-artifact@v4
|
64 |
+
with:
|
65 |
+
run-id: ${{ github.event.workflow_run.id }}
|
66 |
+
github-token: ${{ secrets.GITHUB_TOKEN }}
|
67 |
+
- name: list artifacts
|
68 |
+
run: ls -R .
|
69 |
+
|
70 |
+
- name: Set wheel name
|
71 |
+
id: set_wheel_name
|
72 |
+
run: |
|
73 |
+
wheel_file=$(find ./gradio-build -maxdepth 1 -type f -name "*.whl" -print -quit)
|
74 |
+
echo "wheel_name=$wheel_file" >> $GITHUB_OUTPUT
|
75 |
+
- name: Upload wheel
|
76 |
+
run: |
|
77 |
+
export AWS_ACCESS_KEY_ID=${{ secrets.PYPI_PREVIEWS_AWS_S3_ACCESS_KEY }}
|
78 |
+
export AWS_SECRET_ACCESS_KEY=${{ secrets.PYPI_PREVIEWS_AWS_S3_SECRET_ACCESS_KEY }}
|
79 |
+
export AWS_DEFAULT_REGION=us-east-1
|
80 |
+
aws s3 cp ${{ steps.set_wheel_name.outputs.wheel_name}} s3://gradio-pypi-previews/${{ needs.changes.outputs.sha }}/
|
81 |
+
|
82 |
+
- name: Set tarball name
|
83 |
+
id: set_tarball_name
|
84 |
+
run: |
|
85 |
+
tar_gz_file=$(find ./js-client-tarball -maxdepth 1 -type f -name "*.tgz" -print -quit)
|
86 |
+
echo "tarball_path=$tar_gz_file" >> $GITHUB_OUTPUT
|
87 |
+
echo "tarball_name=$(basename $tar_gz_file)" >> $GITHUB_OUTPUT
|
88 |
+
- name: Upload JS Client Tarball to S3
|
89 |
+
id: upload_js_tarball
|
90 |
+
continue-on-error: true
|
91 |
+
run: |
|
92 |
+
export AWS_ACCESS_KEY_ID=${{ secrets.NPM_PREVIEWS_AWS_S3_ACCESS_KEY }}
|
93 |
+
export AWS_SECRET_ACCESS_KEY=${{ secrets.NPM_PREVIEWS_AWS_S3_SECRET_ACCESS_KEY }}
|
94 |
+
export AWS_DEFAULT_REGION=us-east-1
|
95 |
+
aws s3 cp ${{ steps.set_tarball_name.outputs.tarball_path }} s3://gradio-npm-previews/${{ needs.changes.outputs.sha }}/
|
96 |
+
echo "js_tarball_url=https://gradio-npm-previews.s3.amazonaws.com/${{ needs.changes.outputs.sha }}/${{ steps.set_tarball_name.outputs.tarball_name }}" >> $GITHUB_OUTPUT
|
97 |
+
- name: Upload Lite to S3
|
98 |
+
id: upload_lite
|
99 |
+
run: |
|
100 |
+
export AWS_ACCESS_KEY_ID=${{ secrets.LITE_PREVIEWS_AWS_S3_ACCESS_KEY }}
|
101 |
+
export AWS_SECRET_ACCESS_KEY=${{ secrets.LITE_PREVIEWS_AWS_S3_SECRET_ACCESS_KEY }}
|
102 |
+
export AWS_DEFAULT_REGION=us-east-1
|
103 |
+
mkdir -p ./gradio-lite-files
|
104 |
+
tar -xzf ./gradio-lite-tar/gradio-lite-*.tgz -C ./gradio-lite-files
|
105 |
+
aws s3 cp ./gradio-lite-files/package/ s3://gradio-lite-previews/${{ needs.changes.outputs.sha }}/ --recursive
|
106 |
+
- name: Install Hub Client Library
|
107 |
+
run: |
|
108 |
+
python3 -m venv venv
|
109 |
+
. venv/bin/activate
|
110 |
+
pip install huggingface-hub==0.23.2
|
111 |
+
# temporary, but ensures the script cannot be modified in a PR
|
112 |
+
- name: Get deploy scripts
|
113 |
+
run: |
|
114 |
+
mkdir scripts
|
115 |
+
curl https://raw.githubusercontent.com/gradio-app/gradio/main/scripts/upload_demo_to_space.py -o scripts/upload_demo_to_space.py
|
116 |
+
curl https://raw.githubusercontent.com/gradio-app/gradio/main/scripts/upload_website_demos.py -o scripts/upload_website_demos.py
|
117 |
+
- name: make dirs
|
118 |
+
run: mkdir -p demo && mv all_demos/* demo/
|
119 |
+
- name: Upload demo to spaces
|
120 |
+
if: github.event.workflow_run.event == 'pull_request'
|
121 |
+
id: upload-demo
|
122 |
+
run: |
|
123 |
+
. venv/bin/activate
|
124 |
+
python scripts/upload_demo_to_space.py all_demos \
|
125 |
+
gradio-pr-deploys/pr-${{ needs.changes.outputs.pr_number }}-all-demos \
|
126 |
+
${{ secrets.SPACES_DEPLOY_TOKEN }} \
|
127 |
+
--gradio-version ${{ needs.changes.outputs.gradio_version }} > url.txt
|
128 |
+
echo "SPACE_URL=$(tail -n 1 url.txt)" >> $GITHUB_OUTPUT
|
129 |
+
- name: Upload Website Demos
|
130 |
+
if: github.event.workflow_run.event == 'workflow_dispatch'
|
131 |
+
id: upload-website-demos
|
132 |
+
run: |
|
133 |
+
. venv/bin/activate
|
134 |
+
python scripts/upload_website_demos.py --AUTH_TOKEN ${{ secrets.WEBSITE_SPACES_DEPLOY_TOKEN }} \
|
135 |
+
--WHEEL_URL https://gradio-pypi-previews.s3.amazonaws.com/${{ needs.changes.outputs.sha }}/ \
|
136 |
+
--CLIENT_URL "gradio-client @ git+https://github.com/gradio-app/gradio@${{ needs.changes.outputs.sha }}#subdirectory=client/python" \
|
137 |
+
--GRADIO_VERSION ${{ needs.changes.outputs.gradio_version }}
|
138 |
+
|
139 |
+
comment-spaces-success:
|
140 |
+
uses: "./.github/workflows/comment-queue.yml"
|
141 |
+
needs: [deploy, changes]
|
142 |
+
if: needs.changes.outputs.should_run == 'true' && needs.deploy.result == 'success'
|
143 |
+
secrets:
|
144 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
145 |
+
with:
|
146 |
+
pr_number: ${{ needs.changes.outputs.pr_number }}
|
147 |
+
message: spaces~success~${{ needs.deploy.outputs.space_url }}
|
148 |
+
additional_text: |
|
149 |
+
**Install Gradio from this PR**
|
150 |
+
```bash
|
151 |
+
pip install https://gradio-pypi-previews.s3.amazonaws.com/${{ needs.changes.outputs.sha }}/gradio-${{ needs.changes.outputs.gradio_version }}-py3-none-any.whl
|
152 |
+
```
|
153 |
+
|
154 |
+
**Install Gradio Python Client from this PR**
|
155 |
+
```bash
|
156 |
+
pip install "gradio-client @ git+https://github.com/gradio-app/gradio@${{ needs.changes.outputs.sha }}#subdirectory=client/python"
|
157 |
+
```
|
158 |
+
|
159 |
+
**Install Gradio JS Client from this PR**
|
160 |
+
```bash
|
161 |
+
npm install ${{ needs.deploy.outputs.js_tarball_url }}
|
162 |
+
```
|
163 |
+
|
164 |
+
**Use Lite from this PR**
|
165 |
+
```html
|
166 |
+
<script type="module" src="https://gradio-lite-previews.s3.amazonaws.com/${{ needs.changes.outputs.sha }}/dist/lite.js""></script>
|
167 |
+
```
|
168 |
+
comment-spaces-failure:
|
169 |
+
uses: "./.github/workflows/comment-queue.yml"
|
170 |
+
needs: [deploy, changes]
|
171 |
+
if: always() && needs.deploy.result == 'failure' && needs.changes.outputs.should_run == 'true'
|
172 |
+
secrets:
|
173 |
+
gh_token: ${{ secrets.COMMENT_TOKEN }}
|
174 |
+
with:
|
175 |
+
pr_number: ${{ needs.changes.outputs.pr_number }}
|
176 |
+
message: spaces~failure~https://github.com/gradio-app/gradio/actions/runs/${{github.run_id}}/
|
.github/workflows/publish.yml
ADDED
@@ -0,0 +1,80 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# safe runs from main
|
2 |
+
|
3 |
+
name: publish
|
4 |
+
on:
|
5 |
+
push:
|
6 |
+
branches:
|
7 |
+
- main
|
8 |
+
- 5.0-dev
|
9 |
+
|
10 |
+
env:
|
11 |
+
CI: true
|
12 |
+
PNPM_CACHE_FOLDER: .pnpm-store
|
13 |
+
NODE_OPTIONS: "--max-old-space-size=4096"
|
14 |
+
jobs:
|
15 |
+
version_or_publish:
|
16 |
+
runs-on: ubuntu-22.04
|
17 |
+
environment: publish
|
18 |
+
steps:
|
19 |
+
- name: checkout repo
|
20 |
+
uses: actions/checkout@v4
|
21 |
+
with:
|
22 |
+
fetch-depth: 0
|
23 |
+
persist-credentials: false
|
24 |
+
- name: install dependencies
|
25 |
+
uses: "gradio-app/gradio/.github/actions/install-all-deps@main"
|
26 |
+
with:
|
27 |
+
python_version: "3.10"
|
28 |
+
skip_build: "false"
|
29 |
+
- name: Build packages
|
30 |
+
run: |
|
31 |
+
. venv/bin/activate
|
32 |
+
pip install -U build hatch packaging>=23.2 # packaging>=23.2 is needed to build Lite due to https://github.com/pypa/hatch/issues/1381
|
33 |
+
pnpm --filter @gradio/client --filter @gradio/lite --filter @gradio/preview build
|
34 |
+
|
35 |
+
- name: create and publish versions
|
36 |
+
id: changesets
|
37 |
+
uses: changesets/action@aba318e9165b45b7948c60273e0b72fce0a64eb9 # @v1
|
38 |
+
with:
|
39 |
+
version: pnpm ci:version
|
40 |
+
commit: "chore: update versions"
|
41 |
+
title: "chore: update versions"
|
42 |
+
publish: pnpm ci:publish
|
43 |
+
env:
|
44 |
+
NPM_TOKEN: ${{ secrets.NPM_PUBLISH_TOKEN }}
|
45 |
+
NODE_AUTH_TOKEN: ${{ secrets.NPM_PUBLISH_TOKEN }}
|
46 |
+
GITHUB_TOKEN: ${{ secrets.GRADIO_PAT }}
|
47 |
+
- name: add label to skip chromatic build
|
48 |
+
if: ${{ steps.changesets.outputs.pullRequestNumber != '' && steps.changesets.outputs.pullRequestNumber != 'undefined' }}
|
49 |
+
run: gh pr edit "$PR_NUMBER" --add-label "no-visual-update"
|
50 |
+
env:
|
51 |
+
PR_NUMBER: ${{ steps.changesets.outputs.pullRequestNumber }}
|
52 |
+
GITHUB_TOKEN: ${{ secrets.GRADIO_PAT }}
|
53 |
+
- name: add label to run flaky tests
|
54 |
+
if: ${{ steps.changesets.outputs.pullRequestNumber != '' && steps.changesets.outputs.pullRequestNumber != 'undefined' }}
|
55 |
+
run: gh pr edit "$PR_NUMBER" --add-label "flaky-tests"
|
56 |
+
env:
|
57 |
+
PR_NUMBER: ${{ steps.changesets.outputs.pullRequestNumber }}
|
58 |
+
GITHUB_TOKEN: ${{ secrets.GRADIO_PAT }}
|
59 |
+
- name: add label to run backend tests on Windows
|
60 |
+
if: ${{ steps.changesets.outputs.pullRequestNumber != '' && steps.changesets.outputs.pullRequestNumber != 'undefined' }}
|
61 |
+
run: gh pr edit "$PR_NUMBER" --add-label "windows-tests"
|
62 |
+
env:
|
63 |
+
PR_NUMBER: ${{ steps.changesets.outputs.pullRequestNumber }}
|
64 |
+
GITHUB_TOKEN: ${{ secrets.GRADIO_PAT }}
|
65 |
+
- name: publish to pypi
|
66 |
+
if: steps.changesets.outputs.hasChangesets != 'true'
|
67 |
+
uses: "gradio-app/github/actions/publish-pypi@main"
|
68 |
+
env:
|
69 |
+
AWS_ACCESS_KEY_ID: ${{ secrets.SDK_AWS_S3_BUCKET_ACCESS_KEY }}
|
70 |
+
AWS_SECRET_ACCESS_KEY: ${{ secrets.SDK_AWS_S3_BUCKET_SECRET_ACCESS_KEY }}
|
71 |
+
AWS_DEFAULT_REGION: us-west-2
|
72 |
+
with:
|
73 |
+
user: __token__
|
74 |
+
passwords: |
|
75 |
+
gradio:${{ secrets.PYPI_API_TOKEN }}
|
76 |
+
gradio_client:${{ secrets.PYPI_GRADIO_CLIENT_TOKEN }}
|
77 |
+
- name: trigger spaces deploy workflow
|
78 |
+
env:
|
79 |
+
GITHUB_TOKEN: ${{ secrets.COMMENT_TOKEN }}
|
80 |
+
run: gh workflow run previews-build.yml
|
.github/workflows/semgrep.yml
ADDED
@@ -0,0 +1,68 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: semgrep ci
|
2 |
+
|
3 |
+
on:
|
4 |
+
workflow_run:
|
5 |
+
workflows: ["trigger-semgrep"]
|
6 |
+
types:
|
7 |
+
- completed
|
8 |
+
|
9 |
+
env:
|
10 |
+
CI: true
|
11 |
+
PLAYWRIGHT_SKIP_BROWSER_DOWNLOAD: "1"
|
12 |
+
|
13 |
+
concurrency:
|
14 |
+
group: "${{ github.event.workflow_run.head_repository.full_name }}-${{ github.event.workflow_run.head_branch }}-${{ github.workflow_ref }}"
|
15 |
+
cancel-in-progress: true
|
16 |
+
|
17 |
+
permissions: {}
|
18 |
+
|
19 |
+
jobs:
|
20 |
+
semgrep:
|
21 |
+
permissions:
|
22 |
+
contents: read
|
23 |
+
name: semgrep/ci
|
24 |
+
runs-on: ubuntu-latest
|
25 |
+
container:
|
26 |
+
image: semgrep/semgrep
|
27 |
+
options: --volume ${{ github.workspace }}/.github/configs:/mnt/ --name semgrepcontainer
|
28 |
+
outputs:
|
29 |
+
pr_number: ${{ steps.json.outputs.pr_number }}
|
30 |
+
sha: ${{ steps.json.outputs.sha }}
|
31 |
+
if: (github.actor != 'dependabot[bot]')
|
32 |
+
steps:
|
33 |
+
- name: Download artifact
|
34 |
+
uses: actions/download-artifact@v4
|
35 |
+
with:
|
36 |
+
name: changes
|
37 |
+
github-token: ${{ secrets.GITHUB_TOKEN }}
|
38 |
+
run-id: ${{ github.event.workflow_run.id }}
|
39 |
+
- uses: gradio-app/github/actions/json-to-output@main
|
40 |
+
id: json
|
41 |
+
with:
|
42 |
+
path: output.json
|
43 |
+
- uses: actions/checkout@v4
|
44 |
+
with:
|
45 |
+
repository: ${{ steps.json.outputs.source_repo }}
|
46 |
+
ref: ${{ steps.json.outputs.sha }}
|
47 |
+
- name: restart docker
|
48 |
+
uses: docker://docker
|
49 |
+
with:
|
50 |
+
args: docker restart semgrepcontainer
|
51 |
+
- run: ls -la /mnt
|
52 |
+
- run: semgrep ci --config=/mnt/semgrep_rules.yaml
|
53 |
+
update-status:
|
54 |
+
permissions:
|
55 |
+
actions: read
|
56 |
+
statuses: write
|
57 |
+
runs-on: ubuntu-latest
|
58 |
+
needs: semgrep
|
59 |
+
steps:
|
60 |
+
- name: update status
|
61 |
+
uses: gradio-app/github/actions/commit-status@main
|
62 |
+
with:
|
63 |
+
sha: ${{ needs.semgrep.outputs.sha }}
|
64 |
+
token: ${{ secrets.GITHUB_TOKEN }}
|
65 |
+
name: "Semgrep Results"
|
66 |
+
pr: ${{ needs.semgrep.outputs.pr_number }}
|
67 |
+
result: ${{ needs.semgrep.result == 'success' && 'success' || 'failure' }}
|
68 |
+
type: all
|