Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
xxadev
vscode
提交
6ac86ef0
V
vscode
项目概览
xxadev
/
vscode
与 Fork 源项目一致
从无法访问的项目Fork
通知
2
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
V
vscode
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
提交
6ac86ef0
编写于
5月 21, 2019
作者:
A
Alex Dima
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Implement TunnelService
上级
81b71f06
变更
1
隐藏空白更改
内联
并排
Showing
1 changed file
with
96 addition
and
1 deletion
+96
-1
src/vs/workbench/services/remote/node/tunnelService.ts
src/vs/workbench/services/remote/node/tunnelService.ts
+96
-1
未找到文件。
src/vs/workbench/services/remote/node/tunnelService.ts
浏览文件 @
6ac86ef0
...
...
@@ -3,16 +3,111 @@
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import
*
as
net
from
'
net
'
;
import
{
Barrier
}
from
'
vs/base/common/async
'
;
import
{
Disposable
}
from
'
vs/base/common/lifecycle
'
;
import
{
NodeSocket
}
from
'
vs/base/parts/ipc/node/ipc.net
'
;
import
{
IWorkbenchEnvironmentService
}
from
'
vs/workbench/services/environment/common/environmentService
'
;
import
product
from
'
vs/platform/product/node/product
'
;
import
{
connectRemoteAgentTunnel
,
IConnectionOptions
}
from
'
vs/platform/remote/common/remoteAgentConnection
'
;
import
{
IRemoteAuthorityResolverService
}
from
'
vs/platform/remote/common/remoteAuthorityResolver
'
;
import
{
ITunnelService
,
RemoteTunnel
}
from
'
vs/platform/remote/common/tunnel
'
;
import
{
nodeWebSocketFactory
}
from
'
vs/platform/remote/node/nodeWebSocketFactory
'
;
export
async
function
createRemoteTunnel
(
options
:
IConnectionOptions
,
tunnelRemotePort
:
number
):
Promise
<
RemoteTunnel
>
{
const
tunnel
=
new
NodeRemoteTunnel
(
options
,
tunnelRemotePort
);
return
tunnel
.
waitForReady
();
}
class
NodeRemoteTunnel
extends
Disposable
implements
RemoteTunnel
{
public
readonly
tunnelRemotePort
:
number
;
public
readonly
tunnelLocalPort
:
number
;
private
readonly
_options
:
IConnectionOptions
;
private
readonly
_server
:
net
.
Server
;
private
readonly
_barrier
:
Barrier
;
private
readonly
_listeningListener
:
()
=>
void
;
private
readonly
_connectionListener
:
(
socket
:
net
.
Socket
)
=>
void
;
constructor
(
options
:
IConnectionOptions
,
tunnelRemotePort
:
number
)
{
super
();
this
.
_options
=
options
;
this
.
_server
=
net
.
createServer
();
this
.
_barrier
=
new
Barrier
();
this
.
_listeningListener
=
()
=>
this
.
_barrier
.
open
();
this
.
_server
.
on
(
'
listening
'
,
this
.
_listeningListener
);
this
.
_connectionListener
=
(
socket
)
=>
this
.
_onConnection
(
socket
);
this
.
_server
.
on
(
'
connection
'
,
this
.
_connectionListener
);
this
.
tunnelRemotePort
=
tunnelRemotePort
;
this
.
tunnelLocalPort
=
(
<
net
.
AddressInfo
>
this
.
_server
.
listen
(
0
).
address
()).
port
;
}
public
dispose
():
void
{
super
.
dispose
();
this
.
_server
.
removeListener
(
'
listening
'
,
this
.
_listeningListener
);
this
.
_server
.
removeListener
(
'
connection
'
,
this
.
_connectionListener
);
this
.
_server
.
close
();
}
public
async
waitForReady
():
Promise
<
this
>
{
await
this
.
_barrier
.
wait
();
return
this
;
}
private
async
_onConnection
(
localSocket
:
net
.
Socket
):
Promise
<
void
>
{
// pause reading on the socket until we have a chance to forward its data
localSocket
.
pause
();
const
protocol
=
await
connectRemoteAgentTunnel
(
this
.
_options
,
this
.
tunnelRemotePort
);
const
remoteSocket
=
(
<
NodeSocket
>
protocol
.
getSocket
()).
socket
;
const
dataChunk
=
protocol
.
readEntireBuffer
();
protocol
.
dispose
();
if
(
dataChunk
.
byteLength
>
0
)
{
localSocket
.
write
(
dataChunk
.
buffer
);
}
localSocket
.
on
(
'
end
'
,
()
=>
remoteSocket
.
end
());
localSocket
.
on
(
'
close
'
,
()
=>
remoteSocket
.
end
());
remoteSocket
.
on
(
'
end
'
,
()
=>
localSocket
.
end
());
remoteSocket
.
on
(
'
close
'
,
()
=>
localSocket
.
end
());
localSocket
.
pipe
(
remoteSocket
);
remoteSocket
.
pipe
(
localSocket
);
}
}
export
class
TunnelService
implements
ITunnelService
{
_serviceBrand
:
any
;
public
constructor
(
@
IWorkbenchEnvironmentService
private
readonly
environmentService
:
IWorkbenchEnvironmentService
,
@
IRemoteAuthorityResolverService
private
readonly
remoteAuthorityResolverService
:
IRemoteAuthorityResolverService
,
)
{
}
openTunnel
(
remotePort
:
number
):
Promise
<
RemoteTunnel
>
|
undefined
{
return
undefined
;
const
remoteAuthority
=
this
.
environmentService
.
configuration
.
remoteAuthority
;
if
(
!
remoteAuthority
)
{
return
undefined
;
}
const
options
:
IConnectionOptions
=
{
isBuilt
:
this
.
environmentService
.
isBuilt
,
commit
:
product
.
commit
,
webSocketFactory
:
nodeWebSocketFactory
,
addressProvider
:
{
getAddress
:
async
()
=>
{
const
{
host
,
port
}
=
await
this
.
remoteAuthorityResolverService
.
resolveAuthority
(
remoteAuthority
);
return
{
host
,
port
};
}
}
};
return
createRemoteTunnel
(
options
,
remotePort
);
}
}
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录