ssh-picker 0.1.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 +68 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +192 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/config/paths.d.ts +17 -0
- package/dist/config/paths.js +47 -0
- package/dist/config/paths.js.map +1 -0
- package/dist/db/connection.d.ts +4 -0
- package/dist/db/connection.js +17 -0
- package/dist/db/connection.js.map +1 -0
- package/dist/db/migrations.d.ts +4 -0
- package/dist/db/migrations.js +45 -0
- package/dist/db/migrations.js.map +1 -0
- package/dist/db/repositories/serverRepository.d.ts +13 -0
- package/dist/db/repositories/serverRepository.js +83 -0
- package/dist/db/repositories/serverRepository.js.map +1 -0
- package/dist/db/repositories/settingsRepository.d.ts +8 -0
- package/dist/db/repositories/settingsRepository.js +20 -0
- package/dist/db/repositories/settingsRepository.js.map +1 -0
- package/dist/import-export/archive.d.ts +3 -0
- package/dist/import-export/archive.js +38 -0
- package/dist/import-export/archive.js.map +1 -0
- package/dist/sftp/client.d.ts +21 -0
- package/dist/sftp/client.js +123 -0
- package/dist/sftp/client.js.map +1 -0
- package/dist/shared/credentials.d.ts +2 -0
- package/dist/shared/credentials.js +9 -0
- package/dist/shared/credentials.js.map +1 -0
- package/dist/shared/errors.d.ts +17 -0
- package/dist/shared/errors.js +36 -0
- package/dist/shared/errors.js.map +1 -0
- package/dist/shared/types.d.ts +56 -0
- package/dist/shared/types.js +2 -0
- package/dist/shared/types.js.map +1 -0
- package/dist/ssh/client.d.ts +6 -0
- package/dist/ssh/client.js +45 -0
- package/dist/ssh/client.js.map +1 -0
- package/dist/tui/App.d.ts +5 -0
- package/dist/tui/App.js +27 -0
- package/dist/tui/App.js.map +1 -0
- package/dist/tui/screens/Dashboard.d.ts +8 -0
- package/dist/tui/screens/Dashboard.js +42 -0
- package/dist/tui/screens/Dashboard.js.map +1 -0
- package/dist/tui/screens/FileManager.d.ts +7 -0
- package/dist/tui/screens/FileManager.js +120 -0
- package/dist/tui/screens/FileManager.js.map +1 -0
- package/dist/vault/crypto.d.ts +14 -0
- package/dist/vault/crypto.js +43 -0
- package/dist/vault/crypto.js.map +1 -0
- package/dist/vault/vault.d.ts +7 -0
- package/dist/vault/vault.js +67 -0
- package/dist/vault/vault.js.map +1 -0
- package/package.json +47 -0
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, readdirSync, statSync } from 'node:fs';
|
|
2
|
+
import { basename, dirname, join } from 'node:path';
|
|
3
|
+
import { Client } from 'ssh2';
|
|
4
|
+
import { SshpError } from '../shared/errors.js';
|
|
5
|
+
export function listLocalDirectory(path) {
|
|
6
|
+
return readdirSync(path, { withFileTypes: true })
|
|
7
|
+
.map((entry) => {
|
|
8
|
+
const fullPath = join(path, entry.name);
|
|
9
|
+
const stat = statSync(fullPath);
|
|
10
|
+
return { name: entry.name, path: fullPath, isDirectory: entry.isDirectory(), size: stat.size, modifiedAt: stat.mtime };
|
|
11
|
+
})
|
|
12
|
+
.sort((a, b) => Number(b.isDirectory) - Number(a.isDirectory) || a.name.localeCompare(b.name));
|
|
13
|
+
}
|
|
14
|
+
export class SftpClient {
|
|
15
|
+
conn;
|
|
16
|
+
sftp;
|
|
17
|
+
async connect({ server, credentials }) {
|
|
18
|
+
this.conn = new Client();
|
|
19
|
+
await new Promise((resolve, reject) => {
|
|
20
|
+
this.conn.on('ready', () => {
|
|
21
|
+
this.conn.sftp((err, sftp) => {
|
|
22
|
+
if (err)
|
|
23
|
+
reject(err);
|
|
24
|
+
else {
|
|
25
|
+
this.sftp = sftp;
|
|
26
|
+
resolve();
|
|
27
|
+
}
|
|
28
|
+
});
|
|
29
|
+
});
|
|
30
|
+
this.conn.on('error', reject);
|
|
31
|
+
this.conn.connect({
|
|
32
|
+
host: server.host,
|
|
33
|
+
port: server.port,
|
|
34
|
+
username: server.username,
|
|
35
|
+
password: credentials.password,
|
|
36
|
+
privateKey: credentials.privateKey,
|
|
37
|
+
passphrase: credentials.passphrase,
|
|
38
|
+
readyTimeout: 20_000
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
close() {
|
|
43
|
+
this.conn?.end();
|
|
44
|
+
}
|
|
45
|
+
async listRemoteDirectory(path) {
|
|
46
|
+
const sftp = this.requireSftp();
|
|
47
|
+
const rows = await new Promise((resolve, reject) => {
|
|
48
|
+
sftp.readdir(path, (err, list) => err ? reject(err) : resolve(list));
|
|
49
|
+
});
|
|
50
|
+
return rows.map((entry) => ({
|
|
51
|
+
name: entry.filename,
|
|
52
|
+
path: path.replace(/\/$/, '') + '/' + entry.filename,
|
|
53
|
+
isDirectory: entry.attrs.isDirectory(),
|
|
54
|
+
size: entry.attrs.size,
|
|
55
|
+
modifiedAt: new Date(entry.attrs.mtime * 1000)
|
|
56
|
+
})).sort((a, b) => Number(b.isDirectory) - Number(a.isDirectory) || a.name.localeCompare(b.name));
|
|
57
|
+
}
|
|
58
|
+
async uploadFile(localPath, remotePath, overwrite = false) {
|
|
59
|
+
const sftp = this.requireSftp();
|
|
60
|
+
if (!overwrite && await this.remoteExists(remotePath))
|
|
61
|
+
throw new SshpError(`Remote path already exists: ${remotePath}`, 'OVERWRITE_CONFLICT');
|
|
62
|
+
await new Promise((resolve, reject) => {
|
|
63
|
+
sftp.fastPut(localPath, remotePath, (err) => err ? reject(err) : resolve());
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
async downloadFile(remotePath, localPath, overwrite = false) {
|
|
67
|
+
const sftp = this.requireSftp();
|
|
68
|
+
if (!overwrite && existsSync(localPath))
|
|
69
|
+
throw new SshpError(`Local path already exists: ${localPath}`, 'OVERWRITE_CONFLICT');
|
|
70
|
+
mkdirSync(dirname(localPath), { recursive: true });
|
|
71
|
+
await new Promise((resolve, reject) => {
|
|
72
|
+
sftp.fastGet(remotePath, localPath, (err) => err ? reject(err) : resolve());
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
async uploadRecursive(localPath, remotePath, overwrite = false) {
|
|
76
|
+
const stat = statSync(localPath);
|
|
77
|
+
if (!stat.isDirectory())
|
|
78
|
+
return this.uploadFile(localPath, remotePath, overwrite);
|
|
79
|
+
await this.mkdirRemote(remotePath);
|
|
80
|
+
for (const entry of readdirSync(localPath, { withFileTypes: true })) {
|
|
81
|
+
await this.uploadRecursive(join(localPath, entry.name), remotePath.replace(/\/$/, '') + '/' + entry.name, overwrite);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async downloadRecursive(remotePath, localPath, overwrite = false) {
|
|
85
|
+
const entries = await this.listRemoteDirectory(remotePath);
|
|
86
|
+
mkdirSync(localPath, { recursive: true });
|
|
87
|
+
for (const entry of entries) {
|
|
88
|
+
const target = join(localPath, basename(entry.name));
|
|
89
|
+
if (entry.isDirectory)
|
|
90
|
+
await this.downloadRecursive(entry.path, target, overwrite);
|
|
91
|
+
else
|
|
92
|
+
await this.downloadFile(entry.path, target, overwrite);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
async remoteExists(remotePath) {
|
|
96
|
+
const sftp = this.requireSftp();
|
|
97
|
+
return new Promise((resolve) => {
|
|
98
|
+
sftp.stat(remotePath, (err) => resolve(!err));
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
async mkdirRemote(remotePath) {
|
|
102
|
+
const sftp = this.requireSftp();
|
|
103
|
+
await new Promise((resolve) => {
|
|
104
|
+
sftp.mkdir(remotePath, { mode: 0o755 }, () => resolve());
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
requireSftp() {
|
|
108
|
+
if (!this.sftp)
|
|
109
|
+
throw new SshpError('SFTP client is not connected.', 'SFTP_NOT_CONNECTED');
|
|
110
|
+
return this.sftp;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
export async function withSftp(options, fn) {
|
|
114
|
+
const client = new SftpClient();
|
|
115
|
+
await client.connect(options);
|
|
116
|
+
try {
|
|
117
|
+
return await fn(client);
|
|
118
|
+
}
|
|
119
|
+
finally {
|
|
120
|
+
client.close();
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=client.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/sftp/client.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { decryptString } from '../vault/crypto.js';
|
|
2
|
+
export function decryptServerCredentials(server, vault) {
|
|
3
|
+
return {
|
|
4
|
+
password: server.encryptedPassword ? decryptString(server.encryptedPassword, vault.key) : undefined,
|
|
5
|
+
privateKey: server.encryptedPrivateKey ? decryptString(server.encryptedPrivateKey, vault.key) : undefined,
|
|
6
|
+
passphrase: server.encryptedPassphrase ? decryptString(server.encryptedPassphrase, vault.key) : undefined
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=credentials.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"credentials.js","sourceRoot":"","sources":["../../src/shared/credentials.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAGnD,MAAM,UAAU,wBAAwB,CAAC,MAAoB,EAAE,KAAmB;IAChF,OAAO;QACL,QAAQ,EAAE,MAAM,CAAC,iBAAiB,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,iBAAiB,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS;QACnG,UAAU,EAAE,MAAM,CAAC,mBAAmB,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,mBAAmB,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS;QACzG,UAAU,EAAE,MAAM,CAAC,mBAAmB,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,mBAAmB,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS;KAC1G,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
export declare class SshpError extends Error {
|
|
2
|
+
readonly code: string;
|
|
3
|
+
constructor(message: string, code?: string);
|
|
4
|
+
}
|
|
5
|
+
export declare class MissingVaultError extends SshpError {
|
|
6
|
+
constructor(dataDir: string);
|
|
7
|
+
}
|
|
8
|
+
export declare class VaultExistsError extends SshpError {
|
|
9
|
+
constructor(dataDir: string);
|
|
10
|
+
}
|
|
11
|
+
export declare class InvalidMasterPasswordError extends SshpError {
|
|
12
|
+
constructor();
|
|
13
|
+
}
|
|
14
|
+
export declare class NotFoundError extends SshpError {
|
|
15
|
+
constructor(entity: string, key: string);
|
|
16
|
+
}
|
|
17
|
+
export declare function toFriendlyMessage(error: unknown): string;
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
export class SshpError extends Error {
|
|
2
|
+
code;
|
|
3
|
+
constructor(message, code = 'SSHP_ERROR') {
|
|
4
|
+
super(message);
|
|
5
|
+
this.code = code;
|
|
6
|
+
this.name = 'SshpError';
|
|
7
|
+
}
|
|
8
|
+
}
|
|
9
|
+
export class MissingVaultError extends SshpError {
|
|
10
|
+
constructor(dataDir) {
|
|
11
|
+
super(`No SSHP vault found in ${dataDir}. Run \`sshp init\` first.`, 'MISSING_VAULT');
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
export class VaultExistsError extends SshpError {
|
|
15
|
+
constructor(dataDir) {
|
|
16
|
+
super(`A SSHP vault already exists in ${dataDir}.`, 'VAULT_EXISTS');
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
export class InvalidMasterPasswordError extends SshpError {
|
|
20
|
+
constructor() {
|
|
21
|
+
super('Master password is incorrect.', 'INVALID_MASTER_PASSWORD');
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
export class NotFoundError extends SshpError {
|
|
25
|
+
constructor(entity, key) {
|
|
26
|
+
super(`${entity} not found: ${key}`, 'NOT_FOUND');
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
export function toFriendlyMessage(error) {
|
|
30
|
+
if (error instanceof SshpError)
|
|
31
|
+
return error.message;
|
|
32
|
+
if (error instanceof Error)
|
|
33
|
+
return error.message;
|
|
34
|
+
return String(error);
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=errors.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/shared/errors.ts"],"names":[],"mappings":"AAAA,MAAM,OAAO,SAAU,SAAQ,KAAK;IACW;IAA7C,YAAY,OAAe,EAAkB,OAAO,YAAY;QAC9D,KAAK,CAAC,OAAO,CAAC,CAAC;QAD4B,SAAI,GAAJ,IAAI,CAAe;QAE9D,IAAI,CAAC,IAAI,GAAG,WAAW,CAAC;IAC1B,CAAC;CACF;AAED,MAAM,OAAO,iBAAkB,SAAQ,SAAS;IAC9C,YAAY,OAAe;QACzB,KAAK,CAAC,0BAA0B,OAAO,4BAA4B,EAAE,eAAe,CAAC,CAAC;IACxF,CAAC;CACF;AAED,MAAM,OAAO,gBAAiB,SAAQ,SAAS;IAC7C,YAAY,OAAe;QACzB,KAAK,CAAC,kCAAkC,OAAO,GAAG,EAAE,cAAc,CAAC,CAAC;IACtE,CAAC;CACF;AAED,MAAM,OAAO,0BAA2B,SAAQ,SAAS;IACvD;QACE,KAAK,CAAC,+BAA+B,EAAE,yBAAyB,CAAC,CAAC;IACpE,CAAC;CACF;AAED,MAAM,OAAO,aAAc,SAAQ,SAAS;IAC1C,YAAY,MAAc,EAAE,GAAW;QACrC,KAAK,CAAC,GAAG,MAAM,eAAe,GAAG,EAAE,EAAE,WAAW,CAAC,CAAC;IACpD,CAAC;CACF;AAED,MAAM,UAAU,iBAAiB,CAAC,KAAc;IAC9C,IAAI,KAAK,YAAY,SAAS;QAAE,OAAO,KAAK,CAAC,OAAO,CAAC;IACrD,IAAI,KAAK,YAAY,KAAK;QAAE,OAAO,KAAK,CAAC,OAAO,CAAC;IACjD,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC;AACvB,CAAC"}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
export type AuthType = 'password' | 'private_key';
|
|
2
|
+
export interface EncryptedValue {
|
|
3
|
+
version: 1;
|
|
4
|
+
algorithm: 'aes-256-gcm';
|
|
5
|
+
iv: string;
|
|
6
|
+
tag: string;
|
|
7
|
+
ciphertext: string;
|
|
8
|
+
}
|
|
9
|
+
export interface VaultContext {
|
|
10
|
+
dataDir: string;
|
|
11
|
+
dbPath: string;
|
|
12
|
+
key: Buffer;
|
|
13
|
+
}
|
|
14
|
+
export interface ServerRecord {
|
|
15
|
+
id: number;
|
|
16
|
+
name: string;
|
|
17
|
+
host: string;
|
|
18
|
+
port: number;
|
|
19
|
+
username: string;
|
|
20
|
+
authType: AuthType;
|
|
21
|
+
encryptedPassword: string | null;
|
|
22
|
+
encryptedPrivateKey: string | null;
|
|
23
|
+
encryptedPassphrase: string | null;
|
|
24
|
+
tags: string | null;
|
|
25
|
+
notes: string | null;
|
|
26
|
+
defaultRemotePath: string | null;
|
|
27
|
+
lastConnectedAt: string | null;
|
|
28
|
+
connectionCount: number;
|
|
29
|
+
createdAt: string;
|
|
30
|
+
updatedAt: string;
|
|
31
|
+
}
|
|
32
|
+
export interface CreateServerInput {
|
|
33
|
+
name: string;
|
|
34
|
+
host: string;
|
|
35
|
+
port: number;
|
|
36
|
+
username: string;
|
|
37
|
+
authType: AuthType;
|
|
38
|
+
encryptedPassword?: string | null;
|
|
39
|
+
encryptedPrivateKey?: string | null;
|
|
40
|
+
encryptedPassphrase?: string | null;
|
|
41
|
+
tags?: string | null;
|
|
42
|
+
notes?: string | null;
|
|
43
|
+
defaultRemotePath?: string | null;
|
|
44
|
+
}
|
|
45
|
+
export interface ServerCredentials {
|
|
46
|
+
password?: string;
|
|
47
|
+
privateKey?: string;
|
|
48
|
+
passphrase?: string;
|
|
49
|
+
}
|
|
50
|
+
export interface DirectoryEntry {
|
|
51
|
+
name: string;
|
|
52
|
+
path: string;
|
|
53
|
+
isDirectory: boolean;
|
|
54
|
+
size: number;
|
|
55
|
+
modifiedAt?: Date;
|
|
56
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/shared/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { ServerCredentials, ServerRecord } from '../shared/types.js';
|
|
2
|
+
export interface SshConnectionOptions {
|
|
3
|
+
server: ServerRecord;
|
|
4
|
+
credentials: ServerCredentials;
|
|
5
|
+
}
|
|
6
|
+
export declare function connectSsh({ server, credentials }: SshConnectionOptions): Promise<void>;
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { Client } from 'ssh2';
|
|
2
|
+
export function connectSsh({ server, credentials }) {
|
|
3
|
+
return new Promise((resolve, reject) => {
|
|
4
|
+
const conn = new Client();
|
|
5
|
+
const cleanup = () => {
|
|
6
|
+
if (process.stdin.isTTY)
|
|
7
|
+
process.stdin.setRawMode?.(false);
|
|
8
|
+
process.stdin.unpipe();
|
|
9
|
+
};
|
|
10
|
+
conn.on('ready', () => {
|
|
11
|
+
conn.shell((err, stream) => {
|
|
12
|
+
if (err) {
|
|
13
|
+
conn.end();
|
|
14
|
+
reject(err);
|
|
15
|
+
return;
|
|
16
|
+
}
|
|
17
|
+
if (process.stdin.isTTY)
|
|
18
|
+
process.stdin.setRawMode?.(true);
|
|
19
|
+
process.stdin.resume();
|
|
20
|
+
process.stdin.pipe(stream);
|
|
21
|
+
stream.pipe(process.stdout);
|
|
22
|
+
stream.stderr.pipe(process.stderr);
|
|
23
|
+
stream.on('close', () => {
|
|
24
|
+
cleanup();
|
|
25
|
+
conn.end();
|
|
26
|
+
resolve();
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
});
|
|
30
|
+
conn.on('error', (error) => {
|
|
31
|
+
cleanup();
|
|
32
|
+
reject(error);
|
|
33
|
+
});
|
|
34
|
+
conn.connect({
|
|
35
|
+
host: server.host,
|
|
36
|
+
port: server.port,
|
|
37
|
+
username: server.username,
|
|
38
|
+
password: credentials.password,
|
|
39
|
+
privateKey: credentials.privateKey,
|
|
40
|
+
passphrase: credentials.passphrase,
|
|
41
|
+
readyTimeout: 20_000
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=client.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/ssh/client.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAQ9B,MAAM,UAAU,UAAU,CAAC,EAAE,MAAM,EAAE,WAAW,EAAwB;IACtE,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,MAAM,IAAI,GAAG,IAAI,MAAM,EAAE,CAAC;QAC1B,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,IAAI,OAAO,CAAC,KAAK,CAAC,KAAK;gBAAE,OAAO,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC,KAAK,CAAC,CAAC;YAC3D,OAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;QACzB,CAAC,CAAC;QAEF,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;YACpB,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,MAAM,EAAE,EAAE;gBACzB,IAAI,GAAG,EAAE,CAAC;oBACR,IAAI,CAAC,GAAG,EAAE,CAAC;oBACX,MAAM,CAAC,GAAG,CAAC,CAAC;oBACZ,OAAO;gBACT,CAAC;gBACD,IAAI,OAAO,CAAC,KAAK,CAAC,KAAK;oBAAE,OAAO,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,CAAC;gBAC1D,OAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;gBACvB,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC3B,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;gBAC5B,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;gBACnC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;oBACtB,OAAO,EAAE,CAAC;oBACV,IAAI,CAAC,GAAG,EAAE,CAAC;oBACX,OAAO,EAAE,CAAC;gBACZ,CAAC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,EAAE;YACzB,OAAO,EAAE,CAAC;YACV,MAAM,CAAC,KAAK,CAAC,CAAC;QAChB,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,OAAO,CAAC;YACX,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,QAAQ,EAAE,MAAM,CAAC,QAAQ;YACzB,QAAQ,EAAE,WAAW,CAAC,QAAQ;YAC9B,UAAU,EAAE,WAAW,CAAC,UAAU;YAClC,UAAU,EAAE,WAAW,CAAC,UAAU;YAClC,YAAY,EAAE,MAAM;SACrB,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC"}
|
package/dist/tui/App.js
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { useState } from 'react';
|
|
3
|
+
import { Text } from 'ink';
|
|
4
|
+
import { openVaultDatabase } from '../vault/vault.js';
|
|
5
|
+
import { ServerRepository } from '../db/repositories/serverRepository.js';
|
|
6
|
+
import { Dashboard } from './screens/Dashboard.js';
|
|
7
|
+
import { FileManager } from './screens/FileManager.js';
|
|
8
|
+
import { decryptServerCredentials } from '../shared/credentials.js';
|
|
9
|
+
import { connectSsh } from '../ssh/client.js';
|
|
10
|
+
import { toFriendlyMessage } from '../shared/errors.js';
|
|
11
|
+
export function App({ vault }) {
|
|
12
|
+
const [screen, setScreen] = useState('dashboard');
|
|
13
|
+
const [selectedServer, setSelectedServer] = useState(null);
|
|
14
|
+
const [status, setStatus] = useState(null);
|
|
15
|
+
const db = openVaultDatabase(vault);
|
|
16
|
+
const servers = new ServerRepository(db).list();
|
|
17
|
+
db.close();
|
|
18
|
+
if (screen === 'files' && selectedServer) {
|
|
19
|
+
return _jsx(FileManager, { server: selectedServer, vault: vault, onBack: () => setScreen('dashboard') });
|
|
20
|
+
}
|
|
21
|
+
return _jsxs(_Fragment, { children: [_jsx(Dashboard, { servers: servers, onConnect: (server) => {
|
|
22
|
+
setStatus(`Connecting to ${server.name}...`);
|
|
23
|
+
connectSsh({ server, credentials: decryptServerCredentials(server, vault) })
|
|
24
|
+
.catch((error) => setStatus(toFriendlyMessage(error)));
|
|
25
|
+
}, onFiles: (server) => { setSelectedServer(server); setScreen('files'); } }), status ? _jsx(Text, { color: "yellow", children: status }) : null] });
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=App.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"App.js","sourceRoot":"","sources":["../../src/tui/App.tsx"],"names":[],"mappings":";AAAA,OAAc,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AACxC,OAAO,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAC3B,OAAO,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AACtD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wCAAwC,CAAC;AAE1E,OAAO,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAC;AACpE,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAMxD,MAAM,UAAU,GAAG,CAAC,EAAE,KAAK,EAAY;IACrC,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAwB,WAAW,CAAC,CAAC;IACzE,MAAM,CAAC,cAAc,EAAE,iBAAiB,CAAC,GAAG,QAAQ,CAAsB,IAAI,CAAC,CAAC;IAChF,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAgB,IAAI,CAAC,CAAC;IAC1D,MAAM,EAAE,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAC;IACpC,MAAM,OAAO,GAAG,IAAI,gBAAgB,CAAC,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAChD,EAAE,CAAC,KAAK,EAAE,CAAC;IAEX,IAAI,MAAM,KAAK,OAAO,IAAI,cAAc,EAAE,CAAC;QACzC,OAAO,KAAC,WAAW,IAAC,MAAM,EAAE,cAAc,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,SAAS,CAAC,WAAW,CAAC,GAAI,CAAC;IACrG,CAAC;IAED,OAAO,8BACL,KAAC,SAAS,IACR,OAAO,EAAE,OAAO,EAChB,SAAS,EAAE,CAAC,MAAM,EAAE,EAAE;oBACpB,SAAS,CAAC,iBAAiB,MAAM,CAAC,IAAI,KAAK,CAAC,CAAC;oBAC7C,UAAU,CAAC,EAAE,MAAM,EAAE,WAAW,EAAE,wBAAwB,CAAC,MAAM,EAAE,KAAK,CAAC,EAAE,CAAC;yBACzE,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAC3D,CAAC,EACD,OAAO,EAAE,CAAC,MAAM,EAAE,EAAE,GAAG,iBAAiB,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,GACvE,EACD,MAAM,CAAC,CAAC,CAAC,KAAC,IAAI,IAAC,KAAK,EAAC,QAAQ,YAAE,MAAM,GAAQ,CAAC,CAAC,CAAC,IAAI,IACpD,CAAC;AACN,CAAC"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import type { ServerRecord } from '../../shared/types.js';
|
|
2
|
+
export interface DashboardProps {
|
|
3
|
+
servers: ServerRecord[];
|
|
4
|
+
onConnect: (server: ServerRecord) => void;
|
|
5
|
+
onFiles: (server: ServerRecord) => void;
|
|
6
|
+
onQuit?: () => void;
|
|
7
|
+
}
|
|
8
|
+
export declare function Dashboard({ servers, onConnect, onFiles, onQuit }: DashboardProps): import("react/jsx-runtime").JSX.Element;
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { useMemo, useState } from 'react';
|
|
3
|
+
import { Box, Text, useApp, useInput } from 'ink';
|
|
4
|
+
export function Dashboard({ servers, onConnect, onFiles, onQuit }) {
|
|
5
|
+
const { exit } = useApp();
|
|
6
|
+
const [selected, setSelected] = useState(0);
|
|
7
|
+
const [query, setQuery] = useState('');
|
|
8
|
+
const [searching, setSearching] = useState(false);
|
|
9
|
+
const filtered = useMemo(() => {
|
|
10
|
+
const q = query.trim().toLowerCase();
|
|
11
|
+
return q ? servers.filter((server) => server.name.toLowerCase().includes(q) || server.host.toLowerCase().includes(q)) : servers;
|
|
12
|
+
}, [servers, query]);
|
|
13
|
+
const current = filtered[Math.min(selected, Math.max(0, filtered.length - 1))];
|
|
14
|
+
useInput((input, key) => {
|
|
15
|
+
if (searching) {
|
|
16
|
+
if (key.return || key.escape)
|
|
17
|
+
setSearching(false);
|
|
18
|
+
else if (key.backspace || key.delete)
|
|
19
|
+
setQuery((value) => value.slice(0, -1));
|
|
20
|
+
else if (input)
|
|
21
|
+
setQuery((value) => value + input);
|
|
22
|
+
setSelected(0);
|
|
23
|
+
return;
|
|
24
|
+
}
|
|
25
|
+
if (key.upArrow)
|
|
26
|
+
setSelected((value) => Math.max(0, value - 1));
|
|
27
|
+
if (key.downArrow)
|
|
28
|
+
setSelected((value) => Math.min(Math.max(0, filtered.length - 1), value + 1));
|
|
29
|
+
if (key.return && current)
|
|
30
|
+
onConnect(current);
|
|
31
|
+
if ((input === 'f' || input === 'F') && current)
|
|
32
|
+
onFiles(current);
|
|
33
|
+
if (input === '/')
|
|
34
|
+
setSearching(true);
|
|
35
|
+
if (input === 'q' || input === 'Q' || key.escape) {
|
|
36
|
+
onQuit?.();
|
|
37
|
+
exit();
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
return (_jsxs(Box, { flexDirection: "column", borderStyle: "round", paddingX: 1, children: [_jsxs(Box, { children: [_jsxs(Box, { width: "50%", flexDirection: "column", children: [_jsx(Text, { bold: true, children: "SSHP" }), filtered.length === 0 ? _jsx(Text, { color: "yellow", children: "No servers. Run sshp add." }) : filtered.map((server, index) => (_jsxs(Text, { color: index === selected ? 'cyan' : undefined, children: [index === selected ? '› ' : ' ', server.name, " ", _jsxs(Text, { dimColor: true, children: [server.username, "@", server.host, ":", server.port] })] }, server.id)))] }), _jsxs(Box, { width: "50%", flexDirection: "column", children: [_jsx(Text, { bold: true, children: "Quick Actions" }), _jsx(Text, { children: "Enter Connect SSH" }), _jsx(Text, { children: "F File Manager" }), _jsx(Text, { children: "/ Search" }), _jsx(Text, { children: "Q/Esc Quit" })] })] }), _jsxs(Box, { marginTop: 1, flexDirection: "column", children: [_jsxs(Text, { dimColor: true, children: ["Search: ", searching ? query + '█' : (query || 'press /')] }), _jsx(Text, { dimColor: true, children: "Vault: unlocked" })] })] }));
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=Dashboard.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Dashboard.js","sourceRoot":"","sources":["../../../src/tui/screens/Dashboard.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import type { ServerRecord, VaultContext } from '../../shared/types.js';
|
|
2
|
+
export interface FileManagerProps {
|
|
3
|
+
server: ServerRecord;
|
|
4
|
+
vault: VaultContext;
|
|
5
|
+
onBack: () => void;
|
|
6
|
+
}
|
|
7
|
+
export declare function FileManager({ server, vault, onBack }: FileManagerProps): import("react/jsx-runtime").JSX.Element;
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { useEffect, useState } from 'react';
|
|
3
|
+
import { Box, Text, useApp, useInput } from 'ink';
|
|
4
|
+
import { cwd } from 'node:process';
|
|
5
|
+
import { dirname, join, posix } from 'node:path';
|
|
6
|
+
import { listLocalDirectory, SftpClient } from '../../sftp/client.js';
|
|
7
|
+
import { decryptServerCredentials } from '../../shared/credentials.js';
|
|
8
|
+
import { toFriendlyMessage } from '../../shared/errors.js';
|
|
9
|
+
export function FileManager({ server, vault, onBack }) {
|
|
10
|
+
const { exit } = useApp();
|
|
11
|
+
const [pane, setPane] = useState('local');
|
|
12
|
+
const [localPath, setLocalPath] = useState(cwd());
|
|
13
|
+
const [remotePath, setRemotePath] = useState(server.defaultRemotePath || '.');
|
|
14
|
+
const [localEntries, setLocalEntries] = useState([]);
|
|
15
|
+
const [remoteEntries, setRemoteEntries] = useState([]);
|
|
16
|
+
const [localSelected, setLocalSelected] = useState(0);
|
|
17
|
+
const [remoteSelected, setRemoteSelected] = useState(0);
|
|
18
|
+
const [status, setStatus] = useState('Connecting SFTP...');
|
|
19
|
+
const [client, setClient] = useState(null);
|
|
20
|
+
const [pendingTransfer, setPendingTransfer] = useState(null);
|
|
21
|
+
const refreshLocal = () => {
|
|
22
|
+
try {
|
|
23
|
+
setLocalEntries(listLocalDirectory(localPath));
|
|
24
|
+
}
|
|
25
|
+
catch (error) {
|
|
26
|
+
setStatus(toFriendlyMessage(error));
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
const refreshRemote = async (activeClient = client) => {
|
|
30
|
+
if (!activeClient)
|
|
31
|
+
return;
|
|
32
|
+
try {
|
|
33
|
+
setRemoteEntries(await activeClient.listRemoteDirectory(remotePath));
|
|
34
|
+
setStatus('Ready');
|
|
35
|
+
}
|
|
36
|
+
catch (error) {
|
|
37
|
+
setStatus(toFriendlyMessage(error));
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
useEffect(() => { refreshLocal(); }, [localPath]);
|
|
41
|
+
useEffect(() => {
|
|
42
|
+
const next = new SftpClient();
|
|
43
|
+
next.connect({ server, credentials: decryptServerCredentials(server, vault) })
|
|
44
|
+
.then(() => { setClient(next); setStatus('Ready'); return next.listRemoteDirectory(remotePath); })
|
|
45
|
+
.then(setRemoteEntries)
|
|
46
|
+
.catch((error) => setStatus(toFriendlyMessage(error)));
|
|
47
|
+
return () => next.close();
|
|
48
|
+
}, []);
|
|
49
|
+
useEffect(() => { void refreshRemote(); }, [remotePath, client]);
|
|
50
|
+
useInput((input, key) => {
|
|
51
|
+
const entries = pane === 'local' ? localEntries : remoteEntries;
|
|
52
|
+
const selected = pane === 'local' ? localSelected : remoteSelected;
|
|
53
|
+
const setSelected = pane === 'local' ? setLocalSelected : setRemoteSelected;
|
|
54
|
+
const current = entries[selected];
|
|
55
|
+
if (pendingTransfer) {
|
|
56
|
+
if ((input === 'y' || input === 'Y') && client) {
|
|
57
|
+
const pending = pendingTransfer;
|
|
58
|
+
setPendingTransfer(null);
|
|
59
|
+
setStatus(`${pending.action === 'upload' ? 'Uploading' : 'Downloading'} ${pending.entry.name}...`);
|
|
60
|
+
const task = pending.action === 'upload'
|
|
61
|
+
? client.uploadRecursive(pending.entry.path, pending.target, true).then(() => { setStatus('Upload complete'); void refreshRemote(); })
|
|
62
|
+
: (pending.entry.isDirectory ? client.downloadRecursive(pending.entry.path, pending.target, true) : client.downloadFile(pending.entry.path, pending.target, true))
|
|
63
|
+
.then(() => { setStatus('Download complete'); refreshLocal(); });
|
|
64
|
+
task.catch((error) => setStatus(toFriendlyMessage(error)));
|
|
65
|
+
}
|
|
66
|
+
else if (input === 'n' || input === 'N' || key.escape) {
|
|
67
|
+
setPendingTransfer(null);
|
|
68
|
+
setStatus('Transfer cancelled');
|
|
69
|
+
}
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
if (key.tab)
|
|
73
|
+
setPane((value) => value === 'local' ? 'remote' : 'local');
|
|
74
|
+
if (key.upArrow)
|
|
75
|
+
setSelected((value) => Math.max(0, value - 1));
|
|
76
|
+
if (key.downArrow)
|
|
77
|
+
setSelected((value) => Math.min(Math.max(0, entries.length - 1), value + 1));
|
|
78
|
+
if (key.return && current?.isDirectory) {
|
|
79
|
+
if (pane === 'local') {
|
|
80
|
+
setLocalPath(current.path);
|
|
81
|
+
setLocalSelected(0);
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
setRemotePath(current.path);
|
|
85
|
+
setRemoteSelected(0);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
if (key.backspace) {
|
|
89
|
+
if (pane === 'local')
|
|
90
|
+
setLocalPath(dirname(localPath));
|
|
91
|
+
else
|
|
92
|
+
setRemotePath(posix.dirname(remotePath));
|
|
93
|
+
}
|
|
94
|
+
if (input === 'r' || input === 'R') {
|
|
95
|
+
refreshLocal();
|
|
96
|
+
void refreshRemote();
|
|
97
|
+
}
|
|
98
|
+
if ((input === 'u' || input === 'U') && client && localEntries[localSelected]) {
|
|
99
|
+
const entry = localEntries[localSelected];
|
|
100
|
+
const target = remotePath.replace(/\/$/, '') + '/' + entry.name;
|
|
101
|
+
setPendingTransfer({ action: 'upload', entry, target });
|
|
102
|
+
setStatus(`Upload ${entry.name} to ${target}? Press Y to confirm overwrite, N to cancel.`);
|
|
103
|
+
}
|
|
104
|
+
if ((input === 'd' || input === 'D') && client && remoteEntries[remoteSelected]) {
|
|
105
|
+
const entry = remoteEntries[remoteSelected];
|
|
106
|
+
const target = join(localPath, entry.name);
|
|
107
|
+
setPendingTransfer({ action: 'download', entry, target });
|
|
108
|
+
setStatus(`Download ${entry.name} to ${target}? Press Y to confirm overwrite, N to cancel.`);
|
|
109
|
+
}
|
|
110
|
+
if (input === 'q' || input === 'Q' || key.escape) {
|
|
111
|
+
onBack();
|
|
112
|
+
exit();
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
return (_jsxs(Box, { flexDirection: "column", borderStyle: "round", paddingX: 1, children: [_jsxs(Text, { bold: true, children: ["Files: ", server.name] }), _jsxs(Text, { children: ["Local: ", localPath] }), _jsxs(Text, { children: ["Remote: ", remotePath] }), _jsxs(Box, { marginTop: 1, children: [_jsx(PaneView, { title: "Local files", active: pane === 'local', entries: localEntries, selected: localSelected }), _jsx(PaneView, { title: "Remote files", active: pane === 'remote', entries: remoteEntries, selected: remoteSelected })] }), _jsx(Text, { dimColor: true, children: "Tab pane Enter open Backspace up U upload D download R refresh Q quit" }), _jsx(Text, { color: status === 'Ready' ? 'green' : 'yellow', children: status })] }));
|
|
116
|
+
}
|
|
117
|
+
function PaneView({ title, active, entries, selected }) {
|
|
118
|
+
return _jsxs(Box, { width: "50%", flexDirection: "column", borderStyle: active ? 'single' : undefined, paddingX: 1, children: [_jsx(Text, { bold: true, color: active ? 'cyan' : undefined, children: title }), entries.slice(0, 15).map((entry, index) => _jsxs(Text, { color: index === selected ? 'cyan' : undefined, children: [index === selected ? '› ' : ' ', entry.name, entry.isDirectory ? '/' : ''] }, entry.path))] });
|
|
119
|
+
}
|
|
120
|
+
//# sourceMappingURL=FileManager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FileManager.js","sourceRoot":"","sources":["../../../src/tui/screens/FileManager.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export declare const KDF = "scrypt";
|
|
2
|
+
export declare const CIPHER = "aes-256-gcm";
|
|
3
|
+
export declare const KEY_LENGTH = 32;
|
|
4
|
+
export declare const SCRYPT_OPTIONS: {
|
|
5
|
+
readonly N: number;
|
|
6
|
+
readonly r: 8;
|
|
7
|
+
readonly p: 1;
|
|
8
|
+
readonly maxmem: number;
|
|
9
|
+
};
|
|
10
|
+
export declare function generateSalt(): string;
|
|
11
|
+
export declare function deriveKey(masterPassword: string, saltBase64: string): Buffer;
|
|
12
|
+
export declare function encryptString(plaintext: string, key: Buffer): string;
|
|
13
|
+
export declare function decryptString(serialized: string, key: Buffer): string;
|
|
14
|
+
export declare function safeEquals(a: string, b: string): boolean;
|