参考 https://www.thegeekstuff.com/2013/11/reverse-ssh-tunnel/

https://www.howtoforge.com/reverse-ssh-tunneling

1. 背景

机器A的ip 192.168.0.100,机器B的ip 123.123.123.123

机器A是一个内网机器,可以访问外网。机器B是一个公网机器。也就是说,机器A可以ssh上机器B,但是机器B不能连接机器A。

另一种情况:机器A虽然也是有公网ip,但是由于安全原因,无法被外网访问。

现在,我们想通过机器B去访问机器A。

假设机器A和B都装有ssh服务端,服务端口分别为1001和1002

2. 方案

  ssh的 -R参数

step1:

首先,在机器A上运行

ssh -R 7000:localhost:1001 B_user@123.123.123.123 -p 1002

这样机器A就主动和机器B建立了一个ssh连接。只要机器该建立连接的tty不关闭,此连接一直存在。当然,我们可以将该连接放入后台,同样保证该连接一直存在。

(使用-fn参数?)

一般来说,机器A连接B只需要ssh username@123.123.123.123 就可以了。所以, -R 7000:localhost:1001 是做什么用呢?正是为了做反向连接。

这个-R参数在连接B机器的同时,对B机器做了这样的请求:在localhost端口7000上监听,并将在此接收到的数据包通过我们这个ssh连接转发给我。B按照该请求执行是ssh协议允许的,在B机器上7000端口监听的仍然是sshd进程,它可以决定将数据包用sshd下的哪个线程进行处理。当然,此后发往B机器上7000端口的数据都会倍sshd进程收到,并由A与B的ssh通道发送给A。

另外,A机器收到这个数据后,认为这个数据是发给1001号端口的,即A机器上的ssh服务。整个过程都是在ssh服务框架下完成的。

step2:

在B机器上任开一个tty,执行

ssh A_user@localhost -p 7000

即可连接到A机器了。这个命令的意思是,使用用户名A_user连接本机的7000端口。由于端口为7000的数据包会被通过step1里建立的连接发送给A机器,所以这个连接请求也就最终发给A机器了。

在B机器上使用lsof -i4 -n 可以看到ssh 再127.0.0.1:7000上监听。

在机器A上使用nmap -A 123.123.123.123/32 -p 7000 探测7000端口是关闭,因为B机器监听的地址是本地地址。

但即使将上面命令的localhost改为123.123.123.123,链接后也发现是在127.0.0.1:7000上监听。

3. 扩展

  另一台可以访问到B机器的机器C,如何通过B机器访问A呢?

       首先,C可以ssh到B,然后再一步ssh到A。

但是,有时候需要再C上直接ssh到A,该怎么办呢?
      方案是在B上使用iptables进行端口转发。
    假设A对B发起反向ssh,B在127.0.0.1:7000上监听, 而我们想在C上通过B的端口7000直接ssh到A,由于C只能连接B的公网ip 123.123.123.123,所以我们需要一个新的端口假设7777.
     我们在B上将发往123.123.123.123:7777的数据转发到127.0.0.1:7000
 iptables -t nat -A PREROUTING -d 123.123.123.123 -p tcp -m tcp --dport 7777 -j DNAT --to-destination 127.0.0.1:7000
 
特别注意: 转发到localhost一般需要手工开启本地路由,将/proc/sys/net/ipv4/conf/all/route_localnet写入1
        参考https://superuser.com/questions/661772/iptables-redirect-to-localhost
 

-----------

如果在

ssh -R 7000:localhost:22 B_user@123.123.123.123

命令中将localhost改为*或123.123.123.123,使用 lsof -i4 也可以看到在7000端口监听,但是监听的地址仍然是127.0.0.1。 而且此时无法进行反向ssh,提示

ssh_exchange_identification: Connection closed by remote host

最新文章

  1. Java面试连环炮问题收集记录贴【JVM方面】
  2. btrace使用
  3. kafka入门:简介、使用场景、设计原理、主要配置及集群搭建(转)
  4. WPF面试准备
  5. Hadoop开发第4期---分布式安装
  6. 软件工程(FZU2015)增补作业
  7. 我总结的Android编程规范
  8. 学习macos常用的一些快捷键笔记
  9. hdoj 5494 Card Game
  10. 系统调用和中断处理的异同(以Linux MIPS为例)
  11. POJ 2250 Compromise (UVA 531)
  12. [BZOJ4804]欧拉心算
  13. http post 请求详解
  14. centos升级openssh版本
  15. [转]Linux 微擎系统搭建
  16. DevExpress06、Popup Menus、RadialMenu、XtraTabControl、SplitContainerControl、GroupControl
  17. 当Java遇到XML 的邂逅+dom4j
  18. Swift - 3.0 去掉 C 风格循环
  19. JS中split用法和数组中元素的删除
  20. Laravel 5.2 数据库迁移和数据填充

热门文章

  1. 一种调用dll的巧妙方法
  2. Android有关surfaceView又一次创建的问题。
  3. ubuntu16.04下Cmake学习一
  4. 在PHP中,通过filesize函数可以取得文件的大小,文件大小是以字节数表示的。如果要转换文件大小的单位,可以自己定义函数来实现。
  5. Socket的UDP协议在erlang中的实现
  6. testVC.modalPresentationStyle = UIModalPresentationFormSheet; 更改 VC大小
  7. Golang程序性能分析
  8. 2017-01-20_dp测试
  9. java常量池概念 (转)
  10. 九度OJ 1155:鸡兔同笼 (基础题)