Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
68 commits
Select commit Hold shift + click to select a range
fad641b
added DeleteOrgRepos func
karthikbhandary2 Feb 24, 2026
226ace7
Merge branch 'go-gitea:main' into main
karthikbhandary2 Feb 25, 2026
acf9dc0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Feb 26, 2026
e940dd2
Verified and tested the DeleteRepoOrgs
karthikbhandary2 Feb 26, 2026
c319345
ran make generate-swagger
karthikbhandary2 Feb 26, 2026
e4c5c38
error handling and swagger updated
karthikbhandary2 Feb 26, 2026
ed4b335
Removed additional check for permissions in DeleteOrgRepos
karthikbhandary2 Feb 26, 2026
5b819b4
updated DeleteOrgRepos to delete in the background
karthikbhandary2 Feb 27, 2026
9f31e6c
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 6, 2026
a5092e5
added system notice
karthikbhandary2 Mar 6, 2026
20e3fe4
added batch processing and other fixes
karthikbhandary2 Mar 7, 2026
988ec17
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 7, 2026
008415e
added retry logic
karthikbhandary2 Mar 7, 2026
63f9dbf
reverted back to old logic
karthikbhandary2 Mar 7, 2026
204100a
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 7, 2026
9f7deee
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 8, 2026
bef6617
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 8, 2026
ae7b1a1
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 9, 2026
84e3487
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 9, 2026
974def8
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 10, 2026
e29e9bf
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 17, 2026
ce000b0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 20, 2026
95a0d70
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 21, 2026
cff469d
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
84ef882
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
348feff
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 22, 2026
cfcbdca
updated DeleteOrgRepos to use GetRepositoryByID
Mar 22, 2026
a36a3b5
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 23, 2026
aea3d2b
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 23, 2026
02343a2
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 24, 2026
1633be5
Merge branch 'main' into main
karthikbhandary2 Mar 27, 2026
53a1aac
merge conflict fix
Mar 27, 2026
8b494d6
Merge branch 'go-gitea:main' into main
karthikbhandary2 Mar 28, 2026
41a1ed0
Merge branch 'main' into main
karthikbhandary2 Apr 2, 2026
35587c0
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 3, 2026
e402e5f
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 3, 2026
c8b09c8
Merge branch 'main' into main
karthikbhandary2 Apr 4, 2026
88381aa
add test to check if deletes were successful
Apr 4, 2026
0f0ec79
Merge branch 'main' into main
karthikbhandary2 Apr 5, 2026
41cff2f
refactor
wxiaoguang Apr 5, 2026
f723d35
refactor
wxiaoguang Apr 5, 2026
8ed094e
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 6, 2026
76189cc
tests updated and delete endpoint updated
karthikbhandary2 Apr 6, 2026
bc073ce
Merge branch 'main' into main
karthikbhandary2 Apr 6, 2026
81fc298
Merge branch 'main' into main
karthikbhandary2 Apr 7, 2026
9e71545
clean up AI slop
wxiaoguang Apr 7, 2026
115b582
fine tune
wxiaoguang Apr 7, 2026
8017f5d
fine tune
wxiaoguang Apr 7, 2026
bea4c1c
Merge branch 'main' into main
karthikbhandary2 Apr 7, 2026
bbc3491
remove unnecessary test
wxiaoguang Apr 7, 2026
7af8f5f
Merge branch 'main' into main
karthikbhandary2 Apr 8, 2026
a6bcd02
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 8, 2026
ea82743
Merge branch 'go-gitea:main' into main
karthikbhandary2 Apr 20, 2026
d83c602
feat:Render .ipynb files natively
Apr 26, 2026
53cbc73
Add Jupyter Notebook (.ipynb) rendering support
Apr 27, 2026
6002070
ran lint and fmt
Apr 27, 2026
79d496a
addressed lint issue
Apr 27, 2026
40608b2
fix: lint
Apr 27, 2026
fa84ca7
fix: removed jupyter.css from index.css
Apr 27, 2026
ce940d2
Merge branch 'main' into main
karthikbhandary2 Apr 27, 2026
bb0da24
Merge branch 'main' into main
karthikbhandary2 Apr 28, 2026
d967030
fix
Apr 28, 2026
73ffd64
fix: Moved from custom to using marked for rendering
Apr 28, 2026
ab43de0
Merge branch 'main' into main
karthikbhandary2 Apr 29, 2026
8563401
fix
Apr 29, 2026
55a8bc4
Merge branch 'main' into main
karthikbhandary2 Apr 29, 2026
8b912d3
Merge branch 'main' into main
karthikbhandary2 Apr 30, 2026
22c4803
Merge branch 'main' into main
karthikbhandary2 May 1, 2026
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions modules/markup/external/external.go
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,11 @@ func RegisterRenderers() {
},
})

markup.RegisterRenderer(&frontendRenderer{
name: "jupyter-notebook",
patterns: []string{"*.ipynb"},
})

for _, renderer := range setting.ExternalMarkupRenderers {
markup.RegisterRenderer(&Renderer{renderer})
}
Expand Down
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@
"jquery": "4.0.0",
"js-yaml": "4.1.1",
"katex": "0.16.45",
"marked": "18.0.2",
"mermaid": "11.14.0",
"online-3d-viewer": "0.18.0",
"pdfobject": "2.3.1",
Expand Down
10 changes: 10 additions & 0 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

62 changes: 62 additions & 0 deletions tests/e2e/jupyter-render.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
import {env} from 'node:process';
import {expect, test} from '@playwright/test';
import {login, apiCreateRepo, apiCreateFile, assertNoJsError, randomString} from './utils.ts';

test.describe('jupyter notebook rendering', () => {
let repoName: string;
let owner: string;

test.beforeAll(async ({request}) => {
repoName = `e2e-jupyter-${randomString(8)}`;
owner = env.GITEA_TEST_E2E_USER;

await apiCreateRepo(request, {name: repoName});

// Single comprehensive test notebook
const notebook = JSON.stringify({
cells: [
{cell_type: 'markdown', source: ['# Header 1\n', '## Header 2\n', '**bold** *italic* `code`\n', '- List item 1\n', '- List item 2\n', '[link](https://example.com)\n', '| Col1 | Col2 |\n', '|------|------|\n', '| A | B |\n', '```python\ncode block\n```\n', '> blockquote\n', '~~strikethrough~~']},
{cell_type: 'code', execution_count: 1, source: ['print("Hello")'], outputs: [{output_type: 'stream', name: 'stdout', text: ['Hello\n']}]},
{cell_type: 'code', execution_count: 2, source: ['x'], outputs: [{output_type: 'execute_result', data: {'image/png': 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNk+M9QDwADhgGAWjR9awAAAABJRU5ErkJggg=='}}]},
{cell_type: 'code', source: ['# No output'], outputs: []},
{cell_type: 'code', source: ['err'], outputs: [{output_type: 'error', ename: 'ValueError', evalue: 'Test', traceback: ['ValueError: Test']}]},
{cell_type: 'code', source: ['mixed'], outputs: [{output_type: 'stream', name: 'stdout', text: ['text\n']}, {output_type: 'execute_result', data: {'text/html': ['<b>HTML</b>']}}]},
],
metadata: {}, nbformat: 4, nbformat_minor: 5,
});

await apiCreateFile(request, owner, repoName, 'test.ipynb', notebook);
});

test('renders markdown cells', async ({page}) => {
await login(page);
await page.goto(`/${owner}/${repoName}/src/branch/main/test.ipynb`);
await assertNoJsError(page);
const frame = page.frameLocator('iframe.external-render-iframe');
await expect(frame.locator('.cell.markdown h1')).toBeVisible();
await expect(frame.locator('.cell.markdown strong')).toBeVisible();
await expect(frame.locator('.cell.markdown ul li').first()).toBeVisible();
await expect(frame.locator('.cell.markdown table')).toBeVisible();
});

test('renders code cells with outputs', async ({page}) => {
await login(page);
await page.goto(`/${owner}/${repoName}/src/branch/main/test.ipynb`);
await assertNoJsError(page);
await expect(page.frameLocator('iframe.external-render-iframe').locator('.cell.code .output pre').first()).toBeVisible();
});

test('renders image outputs', async ({page}) => {
await login(page);
await page.goto(`/${owner}/${repoName}/src/branch/main/test.ipynb`);
await assertNoJsError(page);
await expect(page.frameLocator('iframe.external-render-iframe').locator('.cell.code .output img')).toBeVisible();
});

test('renders error outputs', async ({page}) => {
await login(page);
await page.goto(`/${owner}/${repoName}/src/branch/main/test.ipynb`);
await assertNoJsError(page);
await expect(page.frameLocator('iframe.external-render-iframe').locator('.error-output')).toBeVisible();
Comment thread
karthikbhandary2 marked this conversation as resolved.
});
});
192 changes: 192 additions & 0 deletions web_src/css/features/jupyter.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,192 @@
/* Gitea styles for Jupyter notebook content */
.jupyter-notebook {
padding: 20px;
background: var(--color-body);
color: var(--color-text);
font-family: inherit;
}

/* Cell containers */
.jupyter-notebook .cell {
margin-bottom: 20px;
}

/* Markdown cells */
.jupyter-notebook .cell.markdown {
background: var(--color-body);
border: none;
}

.jupyter-notebook .cell.markdown .input {
padding: 6px 12px;
line-height: 1.6;
background: var(--color-body);
color: var(--color-text);
margin-left: 110px;
}

.jupyter-notebook .cell.markdown h1,
.jupyter-notebook .cell.markdown h2,
.jupyter-notebook .cell.markdown h3 {
margin-top: 24px;
margin-bottom: 16px;
font-weight: var(--font-weight-semibold);
line-height: 1.25;
color: var(--color-text);
}

.jupyter-notebook .cell.markdown h1 {
font-size: 1.875em;
border-bottom: 1px solid var(--color-secondary-alpha-20);
padding-bottom: 0.3em;
}

.jupyter-notebook .cell.markdown h2 {
font-size: 1.5em;
}

.jupyter-notebook .cell.markdown h3 {
font-size: 1.25em;
}

.jupyter-notebook .cell.markdown p {
margin-top: 0;
margin-bottom: 16px;
}

.jupyter-notebook .cell.markdown code {
padding: 0.2em 0.4em;
margin: 0;
font-size: 85%;
background-color: var(--color-secondary-alpha-20);
border-radius: 3px;
font-family: var(--fonts-monospace);
}

.jupyter-notebook .cell.markdown table {
border-collapse: collapse;
width: 100%;
margin: 16px 0;
font-size: 13px;
}

.jupyter-notebook .cell.markdown table th,
.jupyter-notebook .cell.markdown table td {
border: 1px solid var(--color-secondary);
padding: 6px 13px;
text-align: left;
}

.jupyter-notebook .cell.markdown table th {
background: var(--color-secondary-alpha-20);
font-weight: var(--font-weight-semibold);
}

.jupyter-notebook .cell.markdown table tr:nth-child(even) {
background: var(--color-secondary-alpha-10);
}

/* Code cells */
.jupyter-notebook .cell.code {
background: transparent;
border: none;
}

.jupyter-notebook .cell.code .input-wrapper,
.jupyter-notebook .cell.code .output-wrapper {
display: flex;
align-items: flex-start;
}

.jupyter-notebook .cell.code .prompt {
padding: 10px 10px 10px 0;
color: var(--color-text-light-2);
font-family: var(--fonts-monospace);
font-size: 13px;
white-space: nowrap;
user-select: none;
text-align: right;
width: 100px;
flex-shrink: 0;
}

.jupyter-notebook .cell.code .input {
flex: 1;
background-color: var(--color-code-bg, #f6f8fa);
border: 1px solid var(--color-secondary-alpha-20, #d0d7de);
border-radius: 4px;
min-height: 40px;
}

.jupyter-notebook .cell.code .input pre {
margin: 0;
padding: 10px 16px;
font-family: var(--fonts-monospace, monospace);
font-size: 13px;
line-height: 1.5;
overflow-x: auto;
color: var(--color-text);
background: transparent;
}

.jupyter-notebook .cell.code .input code {
display: block;
font-family: inherit;
}

/* Code outputs */
.jupyter-notebook .cell.code .output {
flex: 1;
background: var(--color-body);
color: var(--color-text);
overflow-x: auto;
min-width: 0;
}

.jupyter-notebook .cell.code .output pre {
margin: 0;
padding: 10px 16px;
font-family: var(--fonts-monospace);
font-size: 13px;
line-height: 1.5;
overflow-x: auto;
color: var(--color-text);
background: var(--color-body);
white-space: pre-wrap;
overflow-wrap: break-word;
}

.jupyter-notebook .cell.code .output table {
border-collapse: collapse;
margin: 10px 16px;
font-size: 13px;
max-width: 100%;
}

.jupyter-notebook .cell.code .output table th,
.jupyter-notebook .cell.code .output table td {
border: 1px solid var(--color-secondary);
padding: 6px 13px;
text-align: left;
}

.jupyter-notebook .cell.code .output table th {
background: var(--color-secondary-alpha-20);
font-weight: var(--font-weight-semibold);
}

.jupyter-notebook .cell.code .output table tr:nth-child(even) {
background: var(--color-secondary-alpha-10);
}

.jupyter-notebook .cell.code .output img {
max-width: 90%;
height: auto;
display: block;
margin: 10px 0;
}

.jupyter-notebook .cell.code .output table img {
margin: 0;
width: auto;
}
1 change: 1 addition & 0 deletions web_src/js/external-render-frontend.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ type LazyLoadFunc = () => Promise<{frontendRender: FrontendRenderFunc}>;
const frontendPlugins: Record<string, LazyLoadFunc> = {
'viewer-3d': () => import('./render/plugins/frontend-viewer-3d.ts'),
'openapi-swagger': () => import('./render/plugins/frontend-openapi-swagger.ts'),
'jupyter-notebook': () => import('./render/plugins/frontend-jupyter-notebook.ts'),
};

class Options implements FrontendRenderOptions {
Expand Down
Loading