mirror of
https://github.com/Eugeny/tabby
synced 2024-12-13 14:52:45 +00:00
parent
c25d4bd768
commit
8e2ffa1654
2 changed files with 77 additions and 25 deletions
|
@ -9,7 +9,7 @@ import { Injector, NgZone } from '@angular/core'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { ConfigService, FileProvidersService, HostAppService, NotificationsService, Platform, PlatformService, wrapPromise, PromptModalComponent, LogService } from 'tabby-core'
|
import { ConfigService, FileProvidersService, HostAppService, NotificationsService, Platform, PlatformService, wrapPromise, PromptModalComponent, LogService } from 'tabby-core'
|
||||||
import { BaseSession } from 'tabby-terminal'
|
import { BaseSession } from 'tabby-terminal'
|
||||||
import { Socket, createConnection } from 'net'
|
import { Socket } from 'net'
|
||||||
import { Client, ClientChannel, SFTPWrapper } from 'ssh2'
|
import { Client, ClientChannel, SFTPWrapper } from 'ssh2'
|
||||||
import { Subject, Observable } from 'rxjs'
|
import { Subject, Observable } from 'rxjs'
|
||||||
import { ProxyCommandStream } from '../services/ssh.service'
|
import { ProxyCommandStream } from '../services/ssh.service'
|
||||||
|
@ -18,6 +18,7 @@ import { promisify } from 'util'
|
||||||
import { SFTPSession } from './sftp'
|
import { SFTPSession } from './sftp'
|
||||||
import { ALGORITHM_BLACKLIST, SSHAlgorithmType, PortForwardType, SSHProfile } from '../api'
|
import { ALGORITHM_BLACKLIST, SSHAlgorithmType, PortForwardType, SSHProfile } from '../api'
|
||||||
import { ForwardedPort } from './forwards'
|
import { ForwardedPort } from './forwards'
|
||||||
|
import { X11Socket } from './x11'
|
||||||
|
|
||||||
const WINDOWS_OPENSSH_AGENT_PIPE = '\\\\.\\pipe\\openssh-ssh-agent'
|
const WINDOWS_OPENSSH_AGENT_PIPE = '\\\\.\\pipe\\openssh-ssh-agent'
|
||||||
|
|
||||||
|
@ -359,41 +360,35 @@ export class SSHSession extends BaseSession {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
this.ssh.on('x11', (details, accept, reject) => {
|
this.ssh.on('x11', async (details, accept, reject) => {
|
||||||
this.logger.info(`Incoming X11 connection from ${details.srcIP}:${details.srcPort}`)
|
this.logger.info(`Incoming X11 connection from ${details.srcIP}:${details.srcPort}`)
|
||||||
const displaySpec = process.env.DISPLAY ?? ':0'
|
const displaySpec = process.env.DISPLAY ?? 'localhost:0'
|
||||||
this.logger.debug(`Trying display ${displaySpec}`)
|
this.logger.debug(`Trying display ${displaySpec}`)
|
||||||
const xHost = displaySpec.split(':')[0]
|
|
||||||
const xDisplay = parseInt(displaySpec.split(':')[1].split('.')[0] || '0')
|
|
||||||
const xPort = xDisplay < 100 ? xDisplay + 6000 : xDisplay
|
|
||||||
|
|
||||||
const socket = displaySpec.startsWith('/') ? createConnection(displaySpec) : new Socket()
|
const socket = new X11Socket()
|
||||||
if (!displaySpec.startsWith('/')) {
|
try {
|
||||||
socket.connect(xPort, xHost)
|
const x11Stream = await socket.connect(displaySpec)
|
||||||
}
|
this.logger.info('Connection forwarded')
|
||||||
socket.on('error', e => {
|
const stream = accept()
|
||||||
|
stream.pipe(x11Stream)
|
||||||
|
x11Stream.pipe(stream)
|
||||||
|
stream.on('close', () => {
|
||||||
|
socket.destroy()
|
||||||
|
})
|
||||||
|
x11Stream.on('close', () => {
|
||||||
|
stream.close()
|
||||||
|
})
|
||||||
|
} catch (e) {
|
||||||
// eslint-disable-next-line @typescript-eslint/no-base-to-string
|
// eslint-disable-next-line @typescript-eslint/no-base-to-string
|
||||||
this.emitServiceMessage(colors.bgRed.black(' X ') + ` Could not connect to the X server: ${e}`)
|
this.emitServiceMessage(colors.bgRed.black(' X ') + ` Could not connect to the X server: ${e}`)
|
||||||
this.emitServiceMessage(` Tabby tried to connect to ${xHost}:${xPort} based on the DISPLAY environment var (${displaySpec})`)
|
this.emitServiceMessage(` Tabby tried to connect to ${JSON.stringify(X11Socket.resolveDisplaySpec(displaySpec))} based on the DISPLAY environment var (${displaySpec})`)
|
||||||
if (process.platform === 'win32') {
|
if (process.platform === 'win32') {
|
||||||
this.emitServiceMessage(' To use X forwarding, you need a local X server, e.g.:')
|
this.emitServiceMessage(' To use X forwarding, you need a local X server, e.g.:')
|
||||||
this.emitServiceMessage(' * VcXsrv: https://sourceforge.net/projects/vcxsrv/')
|
this.emitServiceMessage(' * VcXsrv: https://sourceforge.net/projects/vcxsrv/')
|
||||||
this.emitServiceMessage(' * Xming: https://sourceforge.net/projects/xming/')
|
this.emitServiceMessage(' * Xming: https://sourceforge.net/projects/xming/')
|
||||||
}
|
}
|
||||||
reject()
|
reject()
|
||||||
})
|
}
|
||||||
socket.on('connect', () => {
|
|
||||||
this.logger.info('Connection forwarded')
|
|
||||||
const stream = accept()
|
|
||||||
stream.pipe(socket)
|
|
||||||
socket.pipe(stream)
|
|
||||||
stream.on('close', () => {
|
|
||||||
socket.destroy()
|
|
||||||
})
|
|
||||||
socket.on('close', () => {
|
|
||||||
stream.close()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
57
tabby-ssh/src/session/x11.ts
Normal file
57
tabby-ssh/src/session/x11.ts
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
import { Socket, SocketConnectOpts } from 'net'
|
||||||
|
import { Subject } from 'rxjs'
|
||||||
|
|
||||||
|
export class X11Socket {
|
||||||
|
error$ = new Subject<Error>()
|
||||||
|
private socket: Socket | null = null
|
||||||
|
|
||||||
|
static resolveDisplaySpec (spec: string): SocketConnectOpts {
|
||||||
|
// eslint-disable-next-line prefer-const
|
||||||
|
let [xHost, xDisplay] = /^(.+):(\d+)(?:.(\d+))$/.exec(spec) ?? []
|
||||||
|
if (process.platform === 'win32') {
|
||||||
|
xHost ??= 'localhost'
|
||||||
|
} else {
|
||||||
|
xHost ??= 'unix'
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spec.startsWith('/')) {
|
||||||
|
xHost = spec
|
||||||
|
}
|
||||||
|
|
||||||
|
const display = parseInt(xDisplay || '0')
|
||||||
|
const port = display < 100 ? display + 6000 : display
|
||||||
|
|
||||||
|
if (xHost === 'unix') {
|
||||||
|
xHost = `/tmp/.X11-unix/X${display}`
|
||||||
|
}
|
||||||
|
|
||||||
|
if (xHost.startsWith('/')) {
|
||||||
|
return {
|
||||||
|
path: xHost,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
host: xHost,
|
||||||
|
port: port,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
connect (spec: string): Promise<Socket> {
|
||||||
|
this.socket = new Socket()
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
this.socket!.on('connect', () => {
|
||||||
|
resolve(this.socket!)
|
||||||
|
})
|
||||||
|
this.socket!.on('error', e => {
|
||||||
|
this.error$.next(e)
|
||||||
|
reject(e)
|
||||||
|
})
|
||||||
|
this.socket!.connect(X11Socket.resolveDisplaySpec(spec))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy (): void {
|
||||||
|
this.socket?.destroy()
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue