klpgit 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +46 -0
- package/dist/cli.js.map +1 -0
- package/dist/server.d.ts +5 -0
- package/dist/server.js +170 -0
- package/dist/server.js.map +1 -0
- package/dist/services/fileTree.d.ts +7 -0
- package/dist/services/fileTree.js +76 -0
- package/dist/services/fileTree.js.map +1 -0
- package/dist/services/git.d.ts +24 -0
- package/dist/services/git.js +125 -0
- package/dist/services/git.js.map +1 -0
- package/package.json +54 -0
- package/web/assets/index-sEsHvKKc.css +1 -0
- package/web/assets/index-vT43bnG2.js +157 -0
- package/web/index.html +17 -0
package/README.md
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
# KlpGit
|
|
2
|
+
|
|
3
|
+
Git GUI that opens right from your terminal. Diff viewer, file tree, commit & push in one click.
|
|
4
|
+
|
|
5
|
+
## Install
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
npm install -g klpgit
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Usage
|
|
12
|
+
|
|
13
|
+
```bash
|
|
14
|
+
cd your-project
|
|
15
|
+
klpgit
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
A window opens with your project. Select files, view changes or source code, run code checks, commit and push.
|
|
19
|
+
|
|
20
|
+
Supports Russian and English — choose on first launch.
|
|
21
|
+
|
|
22
|
+
## License
|
|
23
|
+
|
|
24
|
+
MIT
|
package/dist/cli.d.ts
ADDED
package/dist/cli.js
ADDED
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { startServer } from './server.js';
|
|
3
|
+
import { exec } from 'child_process';
|
|
4
|
+
import open from 'open';
|
|
5
|
+
const cwd = process.cwd();
|
|
6
|
+
const port = parseInt(process.env.KLPGIT_PORT || '4219', 10);
|
|
7
|
+
const url = `http://localhost:${port}`;
|
|
8
|
+
process.stdout.write('\x1Bc');
|
|
9
|
+
console.log('');
|
|
10
|
+
console.log(' \x1b[35m\x1b[1mKlpGit\x1b[0m \x1b[90mv0.2.0\x1b[0m');
|
|
11
|
+
console.log('');
|
|
12
|
+
console.log(` \x1b[36m→\x1b[0m ${url}`);
|
|
13
|
+
console.log('');
|
|
14
|
+
const { server } = startServer(cwd, port);
|
|
15
|
+
function openAppWindow() {
|
|
16
|
+
const w = 1280, h = 840;
|
|
17
|
+
const flags = `--app=${url} --window-size=${w},${h} --disable-extensions --new-window`;
|
|
18
|
+
const isWin = process.platform === 'win32';
|
|
19
|
+
const isMac = process.platform === 'darwin';
|
|
20
|
+
if (isWin) {
|
|
21
|
+
exec(`start msedge ${flags}`, (err) => {
|
|
22
|
+
if (err)
|
|
23
|
+
exec(`start chrome ${flags}`, (err2) => {
|
|
24
|
+
if (err2)
|
|
25
|
+
open(url);
|
|
26
|
+
});
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
else if (isMac) {
|
|
30
|
+
exec(`open -na "Google Chrome" --args ${flags}`, (err) => {
|
|
31
|
+
if (err)
|
|
32
|
+
exec(`open -na "Microsoft Edge" --args ${flags}`, (err2) => {
|
|
33
|
+
if (err2)
|
|
34
|
+
open(url);
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
exec(`google-chrome ${flags} 2>/dev/null || chromium-browser ${flags} 2>/dev/null || microsoft-edge ${flags} 2>/dev/null`, (err) => {
|
|
40
|
+
if (err)
|
|
41
|
+
open(url);
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
server.on('listening', openAppWindow);
|
|
46
|
+
//# sourceMappingURL=cli.js.map
|
package/dist/cli.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,IAAI,EAAE,MAAM,eAAe,CAAC;AACrC,OAAO,IAAI,MAAM,MAAM,CAAC;AAExB,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;AAC1B,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,IAAI,MAAM,EAAE,EAAE,CAAC,CAAC;AAC7D,MAAM,GAAG,GAAG,oBAAoB,IAAI,EAAE,CAAC;AAEvC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;AAE9B,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAChB,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;AACpE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAChB,OAAO,CAAC,GAAG,CAAC,sBAAsB,GAAG,EAAE,CAAC,CAAC;AACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAEhB,MAAM,EAAE,MAAM,EAAE,GAAG,WAAW,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;AAE1C,SAAS,aAAa;IACpB,MAAM,CAAC,GAAG,IAAI,EAAE,CAAC,GAAG,GAAG,CAAC;IACxB,MAAM,KAAK,GAAG,SAAS,GAAG,kBAAkB,CAAC,IAAI,CAAC,oCAAoC,CAAC;IACvF,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC;IAC3C,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,KAAK,QAAQ,CAAC;IAE5C,IAAI,KAAK,EAAE,CAAC;QACV,IAAI,CAAC,gBAAgB,KAAK,EAAE,EAAE,CAAC,GAAG,EAAE,EAAE;YACpC,IAAI,GAAG;gBAAE,IAAI,CAAC,gBAAgB,KAAK,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE;oBAC9C,IAAI,IAAI;wBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;SAAM,IAAI,KAAK,EAAE,CAAC;QACjB,IAAI,CAAC,mCAAmC,KAAK,EAAE,EAAE,CAAC,GAAG,EAAE,EAAE;YACvD,IAAI,GAAG;gBAAE,IAAI,CAAC,oCAAoC,KAAK,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE;oBAClE,IAAI,IAAI;wBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,iBAAiB,KAAK,oCAAoC,KAAK,kCAAkC,KAAK,cAAc,EAAE,CAAC,GAAG,EAAE,EAAE;YACjI,IAAI,GAAG;gBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,MAAM,CAAC,EAAE,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC"}
|
package/dist/server.d.ts
ADDED
package/dist/server.js
ADDED
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import express from 'express';
|
|
2
|
+
import { createServer } from 'http';
|
|
3
|
+
import { WebSocketServer, WebSocket } from 'ws';
|
|
4
|
+
import { watch } from 'chokidar';
|
|
5
|
+
import { join, dirname, resolve } from 'path';
|
|
6
|
+
import { readFileSync, statSync } from 'fs';
|
|
7
|
+
import { fileURLToPath } from 'url';
|
|
8
|
+
import { getFullStatus, getDiff, commitAndPush, addFiles, initRepo, removeRemote, hasGitRepo } from './services/git.js';
|
|
9
|
+
import { getFileTree } from './services/fileTree.js';
|
|
10
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
11
|
+
const __dirname = dirname(__filename);
|
|
12
|
+
export function startServer(cwd, port) {
|
|
13
|
+
const app = express();
|
|
14
|
+
app.use(express.json());
|
|
15
|
+
const webDir = join(__dirname, '..', 'web');
|
|
16
|
+
app.use(express.static(webDir));
|
|
17
|
+
app.get('/api/status', async (_req, res) => {
|
|
18
|
+
try {
|
|
19
|
+
const status = await getFullStatus(cwd);
|
|
20
|
+
res.json(status);
|
|
21
|
+
}
|
|
22
|
+
catch (err) {
|
|
23
|
+
res.status(500).json({ error: String(err) });
|
|
24
|
+
}
|
|
25
|
+
});
|
|
26
|
+
app.get('/api/tree', async (_req, res) => {
|
|
27
|
+
try {
|
|
28
|
+
const tree = getFileTree(cwd);
|
|
29
|
+
res.json(tree);
|
|
30
|
+
}
|
|
31
|
+
catch (err) {
|
|
32
|
+
res.status(500).json({ error: String(err) });
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
app.get('/api/diff', async (req, res) => {
|
|
36
|
+
try {
|
|
37
|
+
const file = req.query.file;
|
|
38
|
+
if (!file) {
|
|
39
|
+
res.json({ diff: '' });
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
const diff = await getDiff(cwd, file);
|
|
43
|
+
res.json({ diff });
|
|
44
|
+
}
|
|
45
|
+
catch (err) {
|
|
46
|
+
res.status(500).json({ error: String(err) });
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
app.post('/api/add', async (req, res) => {
|
|
50
|
+
try {
|
|
51
|
+
const { files } = req.body;
|
|
52
|
+
await addFiles(cwd, files || ['.']);
|
|
53
|
+
const status = await getFullStatus(cwd);
|
|
54
|
+
res.json({ ok: true, status });
|
|
55
|
+
}
|
|
56
|
+
catch (err) {
|
|
57
|
+
res.status(500).json({ error: String(err) });
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
app.post('/api/submit', async (req, res) => {
|
|
61
|
+
try {
|
|
62
|
+
const { message, files } = req.body;
|
|
63
|
+
if (!message) {
|
|
64
|
+
res.status(400).json({ error: 'Commit message required' });
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
const result = await commitAndPush(cwd, message, files || ['.']);
|
|
68
|
+
res.json({ ok: true, ...result });
|
|
69
|
+
}
|
|
70
|
+
catch (err) {
|
|
71
|
+
res.status(500).json({ error: String(err) });
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
app.post('/api/init', async (req, res) => {
|
|
75
|
+
try {
|
|
76
|
+
const { remoteUrl } = req.body;
|
|
77
|
+
await initRepo(cwd, remoteUrl);
|
|
78
|
+
res.json({ ok: true });
|
|
79
|
+
}
|
|
80
|
+
catch (err) {
|
|
81
|
+
res.status(500).json({ error: String(err) });
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
app.post('/api/disconnect', async (_req, res) => {
|
|
85
|
+
try {
|
|
86
|
+
await removeRemote(cwd);
|
|
87
|
+
res.json({ ok: true });
|
|
88
|
+
}
|
|
89
|
+
catch (err) {
|
|
90
|
+
res.status(500).json({ error: String(err) });
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
app.get('/api/file', (req, res) => {
|
|
94
|
+
try {
|
|
95
|
+
const filePath = req.query.path;
|
|
96
|
+
if (!filePath) {
|
|
97
|
+
res.status(400).json({ error: 'Path required' });
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
const fullPath = join(cwd, filePath);
|
|
101
|
+
if (!resolve(fullPath).startsWith(resolve(cwd))) {
|
|
102
|
+
res.status(403).json({ error: 'Access denied' });
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
const stats = statSync(fullPath);
|
|
106
|
+
if (stats.size > 1024 * 1024) {
|
|
107
|
+
res.json({ content: '', language: '' });
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
const content = readFileSync(fullPath, 'utf-8');
|
|
111
|
+
const ext = filePath.split('.').pop()?.toLowerCase() || '';
|
|
112
|
+
res.json({ content, language: ext });
|
|
113
|
+
}
|
|
114
|
+
catch (err) {
|
|
115
|
+
res.status(500).json({ error: String(err) });
|
|
116
|
+
}
|
|
117
|
+
});
|
|
118
|
+
app.get('/api/info', (_req, res) => {
|
|
119
|
+
res.json({
|
|
120
|
+
cwd,
|
|
121
|
+
hasGit: hasGitRepo(cwd),
|
|
122
|
+
name: cwd.split(/[\\/]/).pop() || cwd,
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
const server = createServer(app);
|
|
126
|
+
const wss = new WebSocketServer({ server, path: '/ws' });
|
|
127
|
+
const clients = new Set();
|
|
128
|
+
wss.on('connection', (ws) => {
|
|
129
|
+
clients.add(ws);
|
|
130
|
+
ws.on('close', () => clients.delete(ws));
|
|
131
|
+
});
|
|
132
|
+
function broadcast(event, data) {
|
|
133
|
+
const msg = JSON.stringify({ event, data });
|
|
134
|
+
for (const ws of clients) {
|
|
135
|
+
if (ws.readyState === WebSocket.OPEN) {
|
|
136
|
+
ws.send(msg);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
const watcher = watch(cwd, {
|
|
141
|
+
ignored: [
|
|
142
|
+
/(^|[\/\\])\./,
|
|
143
|
+
'**/node_modules/**',
|
|
144
|
+
'**/dist/**',
|
|
145
|
+
'**/build/**',
|
|
146
|
+
'**/.klpgit/**',
|
|
147
|
+
],
|
|
148
|
+
persistent: true,
|
|
149
|
+
ignoreInitial: true,
|
|
150
|
+
depth: 5,
|
|
151
|
+
});
|
|
152
|
+
let debounceTimer = null;
|
|
153
|
+
const notifyChange = () => {
|
|
154
|
+
if (debounceTimer)
|
|
155
|
+
clearTimeout(debounceTimer);
|
|
156
|
+
debounceTimer = setTimeout(async () => {
|
|
157
|
+
try {
|
|
158
|
+
const status = await getFullStatus(cwd);
|
|
159
|
+
broadcast('status', status);
|
|
160
|
+
}
|
|
161
|
+
catch { }
|
|
162
|
+
}, 300);
|
|
163
|
+
};
|
|
164
|
+
watcher.on('add', notifyChange);
|
|
165
|
+
watcher.on('change', notifyChange);
|
|
166
|
+
watcher.on('unlink', notifyChange);
|
|
167
|
+
server.listen(port, () => { });
|
|
168
|
+
return { server, watcher, port };
|
|
169
|
+
}
|
|
170
|
+
//# sourceMappingURL=server.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { readdirSync, statSync, readFileSync, existsSync } from 'fs';
|
|
2
|
+
import { join } from 'path';
|
|
3
|
+
const DEFAULT_IGNORE = new Set([
|
|
4
|
+
'node_modules', '.git', '.klpgit', 'dist', 'build',
|
|
5
|
+
'.next', '.nuxt', 'coverage', '.cache', '__pycache__',
|
|
6
|
+
'.venv', 'venv', '.env', '.idea', '.vscode',
|
|
7
|
+
]);
|
|
8
|
+
function shouldIgnore(name, gitignorePatterns) {
|
|
9
|
+
if (DEFAULT_IGNORE.has(name))
|
|
10
|
+
return true;
|
|
11
|
+
if (name.startsWith('.') && name !== '.gitignore')
|
|
12
|
+
return true;
|
|
13
|
+
for (const p of gitignorePatterns) {
|
|
14
|
+
if (name === p || name.match(new RegExp('^' + p.replace(/\*/g, '.*') + '$')))
|
|
15
|
+
return true;
|
|
16
|
+
}
|
|
17
|
+
return false;
|
|
18
|
+
}
|
|
19
|
+
function readDir(basePath, relPath, gitignorePatterns) {
|
|
20
|
+
const fullPath = join(basePath, relPath);
|
|
21
|
+
let entries;
|
|
22
|
+
try {
|
|
23
|
+
entries = readdirSync(fullPath);
|
|
24
|
+
}
|
|
25
|
+
catch {
|
|
26
|
+
return [];
|
|
27
|
+
}
|
|
28
|
+
const result = [];
|
|
29
|
+
const sorted = entries
|
|
30
|
+
.filter(e => !shouldIgnore(e, gitignorePatterns))
|
|
31
|
+
.sort((a, b) => {
|
|
32
|
+
const aIsDir = statSync(join(fullPath, a)).isDirectory();
|
|
33
|
+
const bIsDir = statSync(join(fullPath, b)).isDirectory();
|
|
34
|
+
if (aIsDir && !bIsDir)
|
|
35
|
+
return -1;
|
|
36
|
+
if (!aIsDir && bIsDir)
|
|
37
|
+
return 1;
|
|
38
|
+
return a.localeCompare(b);
|
|
39
|
+
});
|
|
40
|
+
for (const entry of sorted) {
|
|
41
|
+
const rel = relPath ? `${relPath}/${entry}` : entry;
|
|
42
|
+
const full = join(basePath, rel);
|
|
43
|
+
let isDir;
|
|
44
|
+
try {
|
|
45
|
+
isDir = statSync(full).isDirectory();
|
|
46
|
+
}
|
|
47
|
+
catch {
|
|
48
|
+
continue;
|
|
49
|
+
}
|
|
50
|
+
const node = {
|
|
51
|
+
name: entry,
|
|
52
|
+
path: rel,
|
|
53
|
+
type: isDir ? 'dir' : 'file',
|
|
54
|
+
};
|
|
55
|
+
if (isDir) {
|
|
56
|
+
node.children = readDir(basePath, rel, gitignorePatterns);
|
|
57
|
+
}
|
|
58
|
+
result.push(node);
|
|
59
|
+
}
|
|
60
|
+
return result;
|
|
61
|
+
}
|
|
62
|
+
export function getFileTree(cwd) {
|
|
63
|
+
let gitignorePatterns = [];
|
|
64
|
+
const gitignorePath = join(cwd, '.gitignore');
|
|
65
|
+
if (existsSync(gitignorePath)) {
|
|
66
|
+
try {
|
|
67
|
+
gitignorePatterns = readFileSync(gitignorePath, 'utf-8')
|
|
68
|
+
.split('\n')
|
|
69
|
+
.map(l => l.trim())
|
|
70
|
+
.filter(l => l && !l.startsWith('#'));
|
|
71
|
+
}
|
|
72
|
+
catch { }
|
|
73
|
+
}
|
|
74
|
+
return readDir(cwd, '', gitignorePatterns);
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=fileTree.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fileTree.js","sourceRoot":"","sources":["../../src/services/fileTree.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { type SimpleGit } from 'simple-git';
|
|
2
|
+
export declare function getGit(cwd: string): SimpleGit;
|
|
3
|
+
export declare function hasGitRepo(cwd: string): boolean;
|
|
4
|
+
export declare function getFullStatus(cwd: string): Promise<{
|
|
5
|
+
branch: string;
|
|
6
|
+
remoteUrl: string;
|
|
7
|
+
staged: number;
|
|
8
|
+
modified: number;
|
|
9
|
+
total: number;
|
|
10
|
+
files: {
|
|
11
|
+
path: string;
|
|
12
|
+
status: string;
|
|
13
|
+
}[];
|
|
14
|
+
ahead: number;
|
|
15
|
+
behind: number;
|
|
16
|
+
}>;
|
|
17
|
+
export declare function getDiff(cwd: string, file: string): Promise<string>;
|
|
18
|
+
export declare function addFiles(cwd: string, files: string[]): Promise<void>;
|
|
19
|
+
export declare function commitAndPush(cwd: string, message: string, filesToAdd: string[]): Promise<{
|
|
20
|
+
hash: string;
|
|
21
|
+
branch: string;
|
|
22
|
+
}>;
|
|
23
|
+
export declare function initRepo(cwd: string, remoteUrl?: string): Promise<void>;
|
|
24
|
+
export declare function removeRemote(cwd: string): Promise<void>;
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import simpleGit from 'simple-git';
|
|
2
|
+
import { existsSync } from 'fs';
|
|
3
|
+
import { join } from 'path';
|
|
4
|
+
export function getGit(cwd) {
|
|
5
|
+
return simpleGit({ baseDir: cwd });
|
|
6
|
+
}
|
|
7
|
+
export function hasGitRepo(cwd) {
|
|
8
|
+
return existsSync(join(cwd, '.git'));
|
|
9
|
+
}
|
|
10
|
+
export async function getFullStatus(cwd) {
|
|
11
|
+
const git = getGit(cwd);
|
|
12
|
+
const status = await git.status();
|
|
13
|
+
const branch = status.current || 'main';
|
|
14
|
+
let remoteUrl = '';
|
|
15
|
+
try {
|
|
16
|
+
const remotes = await git.getRemotes(true);
|
|
17
|
+
const origin = remotes.find((r) => r.name === 'origin');
|
|
18
|
+
remoteUrl = origin?.refs?.fetch || origin?.refs?.push || '';
|
|
19
|
+
}
|
|
20
|
+
catch { }
|
|
21
|
+
const files = [];
|
|
22
|
+
const seen = new Set();
|
|
23
|
+
for (const f of status.staged) {
|
|
24
|
+
if (!seen.has(f)) {
|
|
25
|
+
files.push({ path: f, status: 'staged' });
|
|
26
|
+
seen.add(f);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
for (const f of status.modified) {
|
|
30
|
+
if (!seen.has(f)) {
|
|
31
|
+
files.push({ path: f, status: 'modified' });
|
|
32
|
+
seen.add(f);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
for (const f of status.not_added) {
|
|
36
|
+
if (!seen.has(f)) {
|
|
37
|
+
files.push({ path: f, status: 'untracked' });
|
|
38
|
+
seen.add(f);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
for (const f of status.created) {
|
|
42
|
+
if (!seen.has(f)) {
|
|
43
|
+
files.push({ path: f, status: 'untracked' });
|
|
44
|
+
seen.add(f);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
for (const f of status.deleted) {
|
|
48
|
+
if (!seen.has(f)) {
|
|
49
|
+
files.push({ path: f, status: 'deleted' });
|
|
50
|
+
seen.add(f);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return {
|
|
54
|
+
branch,
|
|
55
|
+
remoteUrl,
|
|
56
|
+
staged: status.staged.length,
|
|
57
|
+
modified: files.filter(f => f.status !== 'staged').length,
|
|
58
|
+
total: files.length,
|
|
59
|
+
files,
|
|
60
|
+
ahead: status.ahead,
|
|
61
|
+
behind: status.behind,
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
export async function getDiff(cwd, file) {
|
|
65
|
+
const git = getGit(cwd);
|
|
66
|
+
let unstaged = '';
|
|
67
|
+
let staged = '';
|
|
68
|
+
try {
|
|
69
|
+
unstaged = await git.diff(['--', file]);
|
|
70
|
+
}
|
|
71
|
+
catch { }
|
|
72
|
+
try {
|
|
73
|
+
staged = await git.diff(['--staged', '--', file]);
|
|
74
|
+
}
|
|
75
|
+
catch { }
|
|
76
|
+
return staged || unstaged || '';
|
|
77
|
+
}
|
|
78
|
+
export async function addFiles(cwd, files) {
|
|
79
|
+
const git = getGit(cwd);
|
|
80
|
+
if (files.includes('.')) {
|
|
81
|
+
await git.add('.');
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
await git.add(files);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
export async function commitAndPush(cwd, message, filesToAdd) {
|
|
88
|
+
const git = getGit(cwd);
|
|
89
|
+
if (filesToAdd.length > 0) {
|
|
90
|
+
if (filesToAdd.includes('.')) {
|
|
91
|
+
await git.add('.');
|
|
92
|
+
}
|
|
93
|
+
else {
|
|
94
|
+
await git.add(filesToAdd);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
const result = await git.commit(message);
|
|
98
|
+
const hash = result.commit || '';
|
|
99
|
+
const status = await git.status();
|
|
100
|
+
const branch = status.current || 'main';
|
|
101
|
+
try {
|
|
102
|
+
await git.push('origin', branch);
|
|
103
|
+
}
|
|
104
|
+
catch {
|
|
105
|
+
await git.push(['-u', 'origin', branch]);
|
|
106
|
+
}
|
|
107
|
+
return { hash, branch };
|
|
108
|
+
}
|
|
109
|
+
export async function initRepo(cwd, remoteUrl) {
|
|
110
|
+
const git = getGit(cwd);
|
|
111
|
+
if (!hasGitRepo(cwd)) {
|
|
112
|
+
await git.init();
|
|
113
|
+
}
|
|
114
|
+
if (remoteUrl) {
|
|
115
|
+
try {
|
|
116
|
+
await git.addRemote('origin', remoteUrl);
|
|
117
|
+
}
|
|
118
|
+
catch { }
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
export async function removeRemote(cwd) {
|
|
122
|
+
const git = getGit(cwd);
|
|
123
|
+
await git.removeRemote('origin');
|
|
124
|
+
}
|
|
125
|
+
//# sourceMappingURL=git.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"git.js","sourceRoot":"","sources":["../../src/services/git.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "klpgit",
|
|
3
|
+
"version": "0.2.0",
|
|
4
|
+
"description": "Beautiful web-based Git GUI — run klpgit, get a local dashboard in your browser",
|
|
5
|
+
"main": "dist/cli.js",
|
|
6
|
+
"type": "module",
|
|
7
|
+
"bin": {
|
|
8
|
+
"klpgit": "dist/cli.js"
|
|
9
|
+
},
|
|
10
|
+
"scripts": {
|
|
11
|
+
"build": "tsc",
|
|
12
|
+
"lint": "eslint src --ext .ts",
|
|
13
|
+
"build:frontend": "cd frontend && npm install && npm run build",
|
|
14
|
+
"build:all": "npm run build:frontend && npm run build",
|
|
15
|
+
"start": "node dist/cli.js",
|
|
16
|
+
"dev": "npm run build && node dist/cli.js",
|
|
17
|
+
"prepublishOnly": "npm run build:all"
|
|
18
|
+
},
|
|
19
|
+
"keywords": [
|
|
20
|
+
"git",
|
|
21
|
+
"gui",
|
|
22
|
+
"web",
|
|
23
|
+
"cli",
|
|
24
|
+
"dashboard"
|
|
25
|
+
],
|
|
26
|
+
"author": "",
|
|
27
|
+
"license": "MIT",
|
|
28
|
+
"engines": {
|
|
29
|
+
"node": ">=18.0.0"
|
|
30
|
+
},
|
|
31
|
+
"dependencies": {
|
|
32
|
+
"chokidar": "^3.5.3",
|
|
33
|
+
"express": "^4.18.2",
|
|
34
|
+
"open": "^10.1.0",
|
|
35
|
+
"simple-git": "^3.22.0",
|
|
36
|
+
"ws": "^8.16.0"
|
|
37
|
+
},
|
|
38
|
+
"devDependencies": {
|
|
39
|
+
"@eslint/js": "^10.0.1",
|
|
40
|
+
"@types/express": "^4.17.21",
|
|
41
|
+
"@types/node": "^20.10.0",
|
|
42
|
+
"@types/ws": "^8.5.10",
|
|
43
|
+
"@typescript-eslint/eslint-plugin": "^8.56.0",
|
|
44
|
+
"@typescript-eslint/parser": "^8.56.0",
|
|
45
|
+
"eslint": "^10.0.1",
|
|
46
|
+
"eslint-import-resolver-typescript": "^4.4.4",
|
|
47
|
+
"globals": "^17.3.0",
|
|
48
|
+
"typescript": "^5.3.0"
|
|
49
|
+
},
|
|
50
|
+
"files": [
|
|
51
|
+
"dist",
|
|
52
|
+
"web"
|
|
53
|
+
]
|
|
54
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
*,*:before,*:after{box-sizing:border-box;margin:0;padding:0}:root{--bg: #0d1117;--bg2: #161b22;--bg3: #1c2129;--bg4: #242b35;--border: #30363d;--border2: #3d444d;--text: #e6edf3;--text2: #7d8590;--text3: #484f58;--accent: #a855f7;--accent2: #c084fc;--accent-bg: rgba(168, 85, 247, .1);--green: #3fb950;--green2: #56d364;--green-bg: rgba(63, 185, 80, .1);--green-line: rgba(63, 185, 80, .15);--red: #f85149;--red2: #ff7b72;--red-bg: rgba(248, 81, 73, .1);--red-line: rgba(248, 81, 73, .15);--yellow: #d29922;--yellow-bg: rgba(210, 153, 34, .1);--cyan: #58a6ff;--cyan-bg: rgba(88, 166, 255, .1);--radius: 8px;--radius-sm: 5px;--mono: "JetBrains Mono", "Cascadia Code", "Fira Code", "Consolas", monospace;--ui: "Inter", -apple-system, "Segoe UI", system-ui, sans-serif}body{background:var(--bg);color:var(--text);font-family:var(--ui);font-size:13px;height:100vh;overflow:hidden}::selection{background:var(--accent);color:#fff}::-webkit-scrollbar{width:6px;height:6px}::-webkit-scrollbar-track{background:transparent}::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}::-webkit-scrollbar-thumb:hover{background:var(--border2)}.app{display:flex;flex-direction:column;height:100vh}.center-screen{display:flex;align-items:center;justify-content:center;flex:1}.spinner{width:20px;height:20px;border:2px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .5s linear infinite}.spin{animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.lang-picker{display:flex;align-items:center;justify-content:center;height:100vh;background:var(--bg)}.lang-content{display:flex;flex-direction:column;align-items:center;gap:24px}.lang-logo{font-size:48px;font-weight:800;color:var(--accent);letter-spacing:-1px}.lang-subtitle{color:var(--text2);font-size:15px;font-weight:500}.lang-options{display:flex;gap:12px}.lang-option{display:flex;align-items:center;gap:10px;padding:16px 32px;border-radius:var(--radius);border:1px solid var(--border);background:var(--bg2);color:var(--text);font-size:16px;font-weight:600;cursor:pointer;transition:all .15s;font-family:var(--ui)}.lang-option:hover{border-color:var(--accent);background:var(--accent-bg);transform:translateY(-2px)}.lang-flag{font-size:11px;font-weight:800;background:var(--bg3);padding:4px 8px;border-radius:4px;font-family:var(--mono);color:var(--accent2)}.titlebar{display:flex;align-items:center;height:38px;background:var(--bg);border-bottom:1px solid var(--border);padding:0 16px;-webkit-app-region:drag;-webkit-user-select:none;user-select:none;flex-shrink:0}.tb-dots{display:flex;gap:7px;width:60px}.tb-dot{width:12px;height:12px;border-radius:50%;-webkit-app-region:no-drag}.dot-r{background:#ff5f57}.dot-y{background:#ffbd2e}.dot-g{background:#28c840}.tb-center{flex:1;text-align:center;font-size:12px;color:var(--text2);font-weight:500;letter-spacing:.3px}.tb-right{width:80px;display:flex;justify-content:flex-end;align-items:center;gap:8px}.tb-lang{background:var(--bg3);border:1px solid var(--border);border-radius:4px;padding:2px 6px;font-size:10px;font-weight:800;font-family:var(--mono);color:var(--accent2);cursor:pointer;transition:all .12s;-webkit-app-region:no-drag}.tb-lang:hover{border-color:var(--accent);background:var(--accent-bg)}.tb-live{width:7px;height:7px;border-radius:50%;transition:all .3s}.tb-live.live{background:var(--green);box-shadow:0 0 8px var(--green)}.tb-live.offline{background:var(--red);box-shadow:0 0 8px var(--red)}.toolbar{display:flex;align-items:center;padding:8px 16px;gap:10px;background:var(--bg2);border-bottom:1px solid var(--border);flex-shrink:0}.tool-logo{font-size:15px;font-weight:800;background:linear-gradient(135deg,var(--accent) 0%,var(--accent2) 100%);-webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text;letter-spacing:-.3px}.tool-sep{width:1px;height:20px;background:var(--border)}.tool-branch{display:flex;align-items:center;gap:5px;background:var(--bg3);border:1px solid var(--border);padding:3px 10px;border-radius:14px;font-size:11px;font-family:var(--mono);color:var(--cyan)}.tool-badges{display:flex;gap:6px;margin-left:auto}.badge{padding:2px 10px;border-radius:12px;font-size:10px;font-weight:700;letter-spacing:.3px;display:flex;align-items:center;gap:4px}.badge-staged{background:var(--cyan-bg);color:var(--cyan)}.badge-changed{background:var(--yellow-bg);color:var(--yellow)}.badge-clean{background:var(--green-bg);color:var(--green)}.tool-actions{display:flex;gap:6px;margin-left:8px}.btn{display:inline-flex;align-items:center;gap:5px;padding:5px 12px;border-radius:var(--radius-sm);border:none;font-size:12px;font-weight:600;cursor:pointer;font-family:var(--ui);transition:all .12s}.btn:hover{transform:translateY(-1px)}.btn:active{transform:translateY(0)}.btn-ghost{background:transparent;color:var(--text2);border:1px solid var(--border)}.btn-ghost:hover{background:var(--bg3);color:var(--text)}.btn-primary{background:var(--accent);color:#fff}.btn-primary:hover{background:var(--accent2)}.btn-primary:disabled{opacity:.6;cursor:not-allowed;transform:none}.btn-danger{color:var(--red);border-color:#f851494d}.btn-danger:hover{background:var(--red-bg);color:var(--red);border-color:var(--red)}.main{display:flex;flex:1;overflow:hidden}.sidebar{width:280px;min-width:240px;background:var(--bg2);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow:hidden}.sb-head{padding:10px 14px;display:flex;align-items:center;justify-content:space-between;border-bottom:1px solid var(--border);font-size:11px;font-weight:700;text-transform:uppercase;letter-spacing:.8px;color:var(--text2)}.sb-count{background:var(--accent-bg);color:var(--accent2);padding:1px 7px;border-radius:10px;font-family:var(--mono);font-size:10px;margin-left:6px}.sb-search{padding:8px 10px;border-bottom:1px solid var(--border)}.sb-search input{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius-sm);padding:6px 10px;font-size:12px;color:var(--text);font-family:var(--mono);outline:none;transition:border .15s}.sb-search input:focus{border-color:var(--accent)}.sb-search input::placeholder{color:var(--text3)}.file-list{flex:1;overflow-y:auto;padding:4px 0}.file-list-empty{padding:30px;text-align:center;color:var(--text3);font-size:12px}.fi{display:flex;align-items:center;padding:4px 12px;gap:7px;cursor:pointer;transition:background .08s;-webkit-user-select:none;user-select:none;font-size:12px;font-family:var(--mono);border-left:2px solid transparent}.fi:hover{background:var(--bg3)}.fi.sel{background:var(--accent-bg);border-left-color:var(--accent)}.fi>div,.fi>span{transition:transform .12s ease}.fi.sel>div,.fi.sel>span{transform:translate(2px)}.fi-icon{width:18px;height:18px;flex-shrink:0;display:flex;align-items:center;justify-content:center}.fi-icon svg{width:16px;height:16px}.fi-name{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.fi-dir .fi-name{color:var(--text);font-weight:600}.fi-check{width:14px;height:14px;border:1.5px solid var(--border2);border-radius:3px;display:flex;align-items:center;justify-content:center;flex-shrink:0;transition:all .12s;cursor:pointer;color:transparent}.fi-check.on{border-color:var(--accent);background:var(--accent);color:#fff}.fi-badge{font-size:9px;font-weight:800;padding:1px 5px;border-radius:3px;flex-shrink:0;letter-spacing:.3px}.fi-badge.M{background:var(--yellow-bg);color:var(--yellow)}.fi-badge.U{background:var(--green-bg);color:var(--green)}.fi-badge.D{background:var(--red-bg);color:var(--red)}.fi-badge.S{background:var(--cyan-bg);color:var(--cyan)}.content{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg)}.content-header{display:flex;align-items:center;gap:0;background:var(--bg2);border-bottom:1px solid var(--border);height:38px;flex-shrink:0}.content-tabs{display:flex;height:100%}.content-tab{display:flex;align-items:center;gap:5px;padding:0 14px;height:100%;border:none;background:transparent;color:var(--text2);font-size:12px;font-weight:500;font-family:var(--ui);cursor:pointer;border-bottom:2px solid transparent;transition:all .12s}.content-tab:hover{color:var(--text);background:var(--bg3)}.content-tab.active{color:var(--text);border-bottom-color:var(--accent)}.content-filename{margin-left:auto;padding-right:16px;font-size:12px;font-family:var(--mono);color:var(--cyan);font-weight:500}.content-stats{display:flex;gap:12px;padding:6px 16px;font-size:12px;font-weight:700;font-family:var(--mono);background:var(--bg2);border-bottom:1px solid var(--border)}.stat-add{color:var(--green)}.stat-del{color:var(--red)}.content-body{flex:1;overflow:hidden;display:flex;flex-direction:column}.content-empty{display:flex;align-items:center;justify-content:center;flex:1;flex-direction:column;gap:10px;color:var(--text3)}.content-empty svg{opacity:.15}.content-empty p{font-size:14px;font-weight:500}.content-empty span{font-size:12px}.content-empty.small{padding:40px}.content-empty.small p{font-size:13px}.diff-body{flex:1;overflow-y:auto;font-family:var(--mono);font-size:12px;line-height:1.7}.dl{display:flex;min-height:22px;border-bottom:1px solid rgba(48,54,61,.3)}.dl:hover{background:#ffffff05}.dl-num{width:50px;flex-shrink:0;text-align:right;padding:0 8px;color:var(--text3);-webkit-user-select:none;user-select:none;font-size:11px;border-right:1px solid var(--border);display:flex;align-items:center;justify-content:flex-end}.dl-sign{width:20px;flex-shrink:0;text-align:center;-webkit-user-select:none;user-select:none;font-weight:700;display:flex;align-items:center;justify-content:center}.dl-code{flex:1;padding:0 12px;white-space:pre;overflow-x:auto}.dl-add{background:var(--green-line);border-left:3px solid var(--green2)}.dl-add .dl-sign,.dl-add .dl-code{color:var(--green2)}.dl-del{background:var(--red-line);border-left:3px solid var(--red2)}.dl-del .dl-sign,.dl-del .dl-code{color:var(--red2)}.dl-hunk{background:var(--cyan-bg)}.dl-hunk .dl-code{color:var(--cyan);font-weight:600;font-style:italic}.dl-meta .dl-code{color:var(--text3)}.code-body{flex:1;overflow-y:auto;font-family:var(--mono);font-size:12px;line-height:1.7}.cl{display:flex;min-height:22px}.cl:hover{background:var(--bg3)}.cl-num{width:50px;flex-shrink:0;text-align:right;padding:0 8px;color:var(--text3);-webkit-user-select:none;user-select:none;font-size:11px;border-right:1px solid var(--border);display:flex;align-items:center;justify-content:flex-end}.cl-code{flex:1;padding:0 12px;white-space:pre;overflow-x:auto;color:var(--text)}.hl-keyword{color:#ff7b72}.hl-string{color:#a5d6ff}.hl-comment{color:#8b949e;font-style:italic}.hl-number{color:#79c0ff}.hl-bool{color:#ff7b72}.hl-fn{color:#d2a8ff}.hl-type{color:#ffa657}.hl-tag{color:#7ee787}.hl-attr{color:#79c0ff}.val-body{flex:1;overflow-y:auto;padding:16px}.val-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.val-stat{font-size:12px;color:var(--text2);font-family:var(--mono)}.val-badge{padding:3px 10px;border-radius:12px;font-size:11px;font-weight:700}.val-ok{background:var(--green-bg);color:var(--green)}.val-warn{background:var(--yellow-bg);color:var(--yellow)}.val-list{display:flex;flex-direction:column;gap:6px}.val-item{display:flex;align-items:center;gap:10px;padding:10px 14px;border-radius:var(--radius-sm);font-size:13px;border:1px solid var(--border);background:var(--bg2)}.val-item-icon{display:flex;flex-shrink:0}.val-warning,.val-warning .val-item-icon{color:var(--yellow)}.val-info,.val-info .val-item-icon{color:var(--cyan)}.val-count{font-weight:700;font-family:var(--mono);min-width:20px;color:var(--text)}.val-msg{flex:1;color:var(--text)}.val-lines{font-size:11px;color:var(--text3);font-family:var(--mono)}.overlay{position:fixed;top:0;right:0;bottom:0;left:0;background:#0009;display:flex;align-items:center;justify-content:center;z-index:100;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);animation:fadeIn .12s}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}@keyframes slideUp{0%{transform:translateY(12px);opacity:0}to{transform:translateY(0);opacity:1}}.modal{background:var(--bg2);border:1px solid var(--border);border-radius:var(--radius);width:440px;max-width:92vw;padding:24px;animation:slideUp .15s;box-shadow:0 20px 60px #00000080}.modal h2{font-size:16px;margin-bottom:4px}.modal-desc{color:var(--text2);font-size:12px;margin-bottom:16px}.modal-input{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius-sm);padding:10px 14px;font-size:14px;color:var(--text);font-family:var(--mono);outline:none;transition:border .15s}.modal-input:focus{border-color:var(--accent);box-shadow:0 0 0 3px #a855f71f}.modal-input::placeholder{color:var(--text3)}.modal-actions{display:flex;gap:8px;margin-top:16px;justify-content:flex-end}.modal-option{display:flex;align-items:center;gap:12px;padding:10px 14px;border-radius:var(--radius-sm);cursor:pointer;transition:all .1s;border:1px solid transparent;margin-bottom:6px}.modal-option:hover{background:var(--bg3);border-color:var(--border)}.modal-option.active{background:var(--accent-bg);border-color:var(--accent)}.modal-option-icon{width:32px;height:32px;border-radius:6px;display:flex;align-items:center;justify-content:center;background:var(--bg3);color:var(--accent2)}.modal-option-label{font-weight:600;font-size:13px}.modal-option-desc{color:var(--text2);font-size:11px;margin-top:1px}.toast-container{position:fixed;bottom:16px;right:16px;z-index:200;display:flex;flex-direction:column-reverse;gap:8px}.toast{padding:8px 14px;border-radius:var(--radius-sm);font-size:13px;font-weight:500;animation:slideIn .2s;box-shadow:0 8px 24px #0006;display:flex;align-items:center;gap:8px;cursor:pointer}@keyframes slideIn{0%{transform:translateY(16px);opacity:0}to{transform:translateY(0);opacity:1}}.toast-ok{background:#0d2818;color:var(--green);border:1px solid #1a4028}.toast-error{background:#2a0f0f;color:var(--red);border:1px solid #4a1a1a}.toast-info{background:#0d1f30;color:var(--cyan);border:1px solid #1a3050}.init-screen{display:flex;align-items:center;justify-content:center;flex:1;flex-direction:column;gap:16px}.init-logo{display:flex;flex-direction:column;align-items:center;gap:8px;color:var(--accent)}.init-logo h1{font-size:36px;font-weight:800;letter-spacing:-1px}.init-screen p{color:var(--text2);font-size:14px}.init-screen .modal-input{max-width:380px}
|