pastebin - collaborative debugging tool
rovema.kpaste.net RSS


sshnfs - ssh with nfs forwarding
Posted by Anonymous on Mon 17th Jul 2023 15:25
raw | new post
view followups (newest first): sshnfs - ssh with nfs forwarding by Anonymous
modification of post by Anonymous (view diff)

  1. #!/usr/bin/ksh93
  2.  
  3. #
  4. # sshnfs - remote login client with NFSv4 forwarding
  5. #
  6.  
  7. #
  8. # Example usage:
  9. # $ ksh sshnfs.ksh -o NFSURL=ssh+nfs://localhost/export/home/rmainz root@10.49.28.10 #
  10. # $ ksh sshnfs.ksh -o NFSURL=nfs://localhost/export/home/rmainz root@10.49.20.207 #
  11. # $ ksh sshnfs.ksh -o NFSURL=nfs://derfwpc5131/export/home/rmainz root@10.49.28.10 #
  12. # $ ksh sshnfs.ksh -o NFSURL=ssh+nfs://derfwpc5131/export/home/rmainz -o SSHNFSJumphost=rmainz@derfwpc5131,roland.mainz@derfwnb8353 -J rmainz@derfwpc5131,roland.mainz@derfwnb8353 root@10.49.20.207
  13. # $ ksh sshnfs.ksh -o NFSURL=ssh+nfs://derfwpc5131/export/home/rmainz target@fe80::d6f5:27ff:fe2b:8588%enp2s0
  14. # $ ksh sshnfs.ksh -o NFSServerSSHLoginName=root -o NFSURL=ssh+nfs://derfwpc5131/export/home/rmainz root@10.49.28.56
  15. # $ SSHNFS_OPTIONS='-o NFSServerSSHLoginName=root -o NFSURL=ssh+nfs://derfwpc5131/export/home/rmainz' sshnfs.ksh root@10.49.28.56
  16. #
  17. # Written by Roland Mainz <roland.mainz@nrubsig.org>
  18. #
  19.  
  20. #    
  21. # simple netstat -n parser
  22. #    
  23. function netstat_list_connections
  24. {
  25.         set -o nounset
  26.         nameref data=$1
  27.        
  28.         compound out=( typeset stdout stderr ; integer res )
  29.        
  30.         out.stderr="${ { out.stdout="${ LC_ALL='POSIX' PATH='/usr/bin:/bin' netstat -a -n ; (( out.res=$? )) ; }" ; } 2>&1 ; }"
  31.         if (( out.res != 0 )) || [[ ${out.stderr} != '' ]] ; then
  32.                 return 1
  33.         fi
  34.  
  35.         typeset -a data.connections
  36.         typeset l
  37.         integer dci=0 # data.connections array index
  38.  
  39.         while read l ; do
  40.                 leftover="${l/~(Elrx)
  41.                 (?: # non-capturing group
  42.                         #
  43.                         # regex group for tcp,udp
  44.                         #
  45.                         (tcp|tcp6|udp|udp6|raw|raw6|sctp|sctp6) # Proto
  46.                         [[:space:]]+
  47.                         ([[:digit:]]+)                  # Recv-Q
  48.                         [[:space:]]+
  49.                         ([[:digit:]]+)                  # Send-Q
  50.                         [[:space:]]+
  51.                         ([^[:space:]]+)                 # Local Address
  52.                         [[:space:]]+
  53.                         ([^[:space:]]+)                 # Foreign Address
  54.                         (?:
  55.                                 |
  56.                                 [[:space:]]+
  57.                                 ([^[:space:]]*?)        # State (Optional)
  58.                         )
  59.                 |
  60.                         #
  61.                         # regex for unix
  62.                         #
  63.                         (unix)                          # Proto
  64.                         [[:space:]]+
  65.                         ([[:digit:]]+)                  # RefCnt
  66.                         [[:space:]]+
  67.                         (\[.+?\])                       # Flags
  68.                         [[:space:]]+
  69.                         ([^[:space:]]+)                 # Type
  70.                         [[:space:]]+
  71.                         ([^[:space:]]*?)                # State (optional)
  72.                         [[:space:]]+
  73.                         ([[:digit:]]+)                  # I-Node
  74.                         (?:
  75.                                 |
  76.                                 [[:space:]]+
  77.                                 ([^[:space:]]+)         # Path (optional)
  78.                         )
  79.                 )
  80.                         /X}"
  81.        
  82.                 # If the regex above did not match then .sh.match
  83.                 # remains untouched, so we might see data from the
  84.                 # previous round.
  85.                 # So we check the "leftover" var whether it just
  86.                 # contains the dummy value of "X" to indicate a
  87.                 # successful regex match
  88.                 if [[ "$leftover" == 'X' ]] ; then
  89.                         #print -v .sh.match
  90.                        
  91.                         if [[ "${.sh.match[1]-}" != '' ]] ; then
  92.                                 nameref dcn=data.connections[$dci]
  93.  
  94.                                 typeset dcn.proto="${.sh.match[1]}"
  95.                                 typeset dcn.recv_q="${.sh.match[2]}"
  96.                                 typeset dcn.send_q="${.sh.match[3]}"
  97.                                 typeset dcn.local_address="${.sh.match[4]}"
  98.                                 typeset dcn.foreign_address="${.sh.match[5]}"
  99.                                 typeset dcn.state="${.sh.match[6]}"
  100.                                 ((dci++))
  101.                         elif [[ "${.sh.match[7]-}" != '' ]] ; then
  102.                                 nameref dcn=data.connections[$dci]
  103.  
  104.                                 typeset dcn.proto="${.sh.match[7]}"
  105.                                 typeset dcn.refcnt="${.sh.match[8]}"
  106.                                 typeset dcn.flags="${.sh.match[9]}"
  107.                                 typeset dcn.type="${.sh.match[10]}"
  108.                                 [[ "${.sh.match[11]}" != '' ]] && typeset dcn.state="${.sh.match[11]}"
  109.                                 typeset dcn.inode="${.sh.match[12]}"
  110.                                 [[ "${.sh.match[13]}" != '' ]] && typeset dcn.path="${.sh.match[13]}"
  111.                                 ((dci++))
  112.                         fi
  113.                 else
  114.                         true
  115.                         #printf $"leftover=%q\n" "${leftover}"
  116.                 fi
  117.         done <<<"${out.stdout}"
  118.        
  119.         return 0
  120. }
  121.  
  122. function netstat_list_active_local_tcp_connections
  123. {
  124.         set -o nounset
  125.         nameref ar=$1
  126.         compound c
  127.         integer port
  128.         integer i
  129.  
  130.         netstat_list_connections c || return 1
  131.         #print -v c
  132.        
  133.         [[ -v ar ]] || integer -a ar
  134.  
  135.         for i in "${!c.connections[@]}" ; do
  136.                 nameref n=c.connections[$i]
  137.                
  138.                 # look for only for TCP connections which match
  139.                 # 127.0.*.* or IPv6 ::1 for localhost
  140.                 # 0.0.0.0 or IPv6 :: for all addresses (e.g. servers)
  141.                 if [[ "${n.proto}" == ~(El)tcp && \
  142.                         "${n.local_address}" == ~(Elr)((127\.0\..+|::1)|(::|0\.0\.0\.0|)):[[:digit:]]+ ]] ; then
  143.  
  144.                         port="${n.local_address##*:}"
  145.                         #printf $"port = %d\n" port
  146.  
  147.                         (( ar[port]=1 ))
  148.                 fi
  149.         done
  150.  
  151.         return 0
  152. }
  153.  
  154. function netstat_find_next_free_local_tcp_port
  155. {
  156.         set -o nounset
  157.         compound c=( integer -a ar )
  158.         nameref ret_free_port=$1
  159.         integer start_port
  160.         integer end_port
  161.         integer i
  162.  
  163.         netstat_list_active_local_tcp_connections c.ar || return 1
  164.  
  165.         #print -v c
  166.  
  167.         (( start_port=$2 ))
  168.         if (( $# > 2 )) ; then
  169.                 (( end_port=$3 ))
  170.         else
  171.                 (( end_port=65535 ))
  172.         fi
  173.  
  174.         for ((i=start_port ; i < end_port ; i++ )) ; do
  175.                 if [[ ! -v c.ar[i] ]] ; then
  176.                         (( ret_free_port=i ))
  177.                         return 0
  178.                 fi
  179.         done
  180.        
  181.         return 1
  182. }
  183.  
  184. #
  185. # parse url
  186. #
  187. # returns:
  188. # data.protocol
  189. # data.host
  190. # data.port (optional)
  191. # data.path
  192. #
  193. function parse_url
  194. {
  195.         typeset url="$2"
  196.         typeset leftover
  197.         nameref data="$1"
  198.        
  199.         # ~(E) is POSIX extended regular expression matching (instead of
  200.         # shell pattern)
  201.         leftover="${url//~(Elr)(.+?):\/\/(.+?)(?:|:([[:digit:]]+))(?:\/(.*?))?/X}"
  202.        
  203.         # All parsed data should be captured via eregex in .sh.match - if
  204.         # there is anything left (except the 'X') then the input string did not
  205.         # properly match the eregex
  206.         [[ "$leftover" == 'X' ]] || { print -u2 -f $"%s: Parser error\n" "$0" ; return 1 ; }
  207.  
  208.         data.protocol="${.sh.match[1]}"
  209.         data.host="${.sh.match[2]}"
  210.         # bug: should be [[ -v .sh.match[3] ]], but ksh93u has bugs
  211.         [[ "${.sh.match[3]}" != '' ]] && integer data.port="${.sh.match[3]}"
  212.         data.path="${.sh.match[4]}"
  213.  
  214.         return 0
  215. }
  216.  
  217. function parse_sshnfs_url
  218. {
  219.         typeset url="$2"
  220.         nameref data="$1"
  221.        
  222.         parse_url data "$url" || return 1
  223.        
  224.         [[ "${data.protocol}" == ~(Elr)(ssh\+nfs|nfs) ]] || \
  225.                 { print -u2 -f $"%s: Not a nfs:// or ssh+nfs:// url\n" "$0" ; return 1 ; }
  226.         [[ "${data.host}" != '' ]] || { print -u2 -f $"%s: NFS hostname missing\n" "$0" ; return 1 ; }
  227.         [[ "${data.path}" != '' ]] || { print -u2 -f $"%s: NFS path missing\n" "$0" ; return 1 ; }
  228.        
  229.         if [[ ! -v data.port ]] ; then
  230.                 # use # default NFSv4 TCP port number (see
  231.                 # $ getent services nfs #)
  232.                 integer data.port=2049
  233.         fi
  234.  
  235.         return 0
  236. }
  237.  
  238.  
  239. function main
  240. {
  241.         set -o nounset
  242.         typeset mydebug=false # fixme: should be "bool" for ksh93v
  243.         integer i
  244.         integer retval
  245.         compound c=(
  246.                 # User name for SSH login to NFS server
  247.                 typeset nfsserver_ssh_login_name="$LOGNAME"
  248.         )
  249.  
  250.  
  251.  
  252.         #
  253.         # Expand SSHNFS_OPTIONS before arguments given to sshnfs.ksh
  254.         # By default we use IFS=$' \t\n' for argument splitting
  255.         #
  256.         typeset c.args=( ${SSHNFS_OPTIONS-} "$@" )
  257.  
  258.         for ((i=0 ; i < ${#c.args[@]} ; i++)) ; do
  259.                 if [[ "${c.args[i]}" == '-o' ]] ; then
  260.                         case "${c.args[i+1]-}" in
  261.                                 ~(Eli)NFSServerSSHLoginName=)
  262.                                         c.nfsserver_ssh_login_name="${c.args[i+1]/~(Eli)NFSServerSSHLoginName=}"
  263.  
  264.                                         unset c.args[$i] c.args[$((i+1))]
  265.                                         ((i++))
  266.                                         ;;
  267.  
  268.                                 ~(Eli)NFSURL=)
  269.                                         unset c.nfs_server
  270.                                         compound c.nfs_server
  271.                                         typeset c.url="${c.args[i+1]/~(Eli)NFSURL=}"
  272.                                         parse_sshnfs_url c.nfs_server "${c.url}" || return 1
  273.  
  274.                                         unset c.args[$i] c.args[$((i+1))]
  275.                                         ((i++))
  276.                                         ;;
  277.  
  278.                                 ~(Eli)SSHNFSJumphost=)
  279.                                         [[ ! -v c.ssh_jumphost_args ]] && typeset -a c.ssh_jumphost_args
  280.                                         c.ssh_jumphost_args+=( "-J" "${c.args[i+1]/~(Eli)SSHNFSJumphost=}" )
  281.  
  282.                                         unset c.args[$i] c.args[$((i+1))]
  283.                                         ((i++))
  284.                                         ;;
  285.                         esac
  286.                 fi
  287.         done
  288.  
  289.         if [[ -v c.nfs_server ]] ; then
  290.                 case "${c.nfs_server.protocol}" in
  291.                         'ssh+nfs')
  292.                                 #
  293.                                 # find free local forwarding port...
  294.                                 #
  295.  
  296.                                 # port on THIS machine
  297.                                 integer c.local_forward_port
  298.  
  299.                                 # TCP port on destination machine where we forward the
  300.                                 # NFS port from the server
  301.                                 integer c.destination_nfs_port=33049
  302.  
  303.                                 (( i=34049 ))
  304.                                 netstat_find_next_free_local_tcp_port c.local_forward_port $i
  305.  
  306.                                 #
  307.                                 # ... and adjust c.destination_nfs_port by the same offset
  308.                                 # we do that so that multiple sshweb.ksh logins to the same
  309.                                 # machine do try to use the same ports on that machine
  310.                                 #
  311.                                 (( c.destination_nfs_port += c.local_forward_port-i ))
  312.                                 ${mydebug} && printf $"debug: c.local_forward_port=%d, c.destination_nfs_port=%d\n" \
  313.                                         c.local_forward_port \
  314.                                         c.destination_nfs_port
  315.  
  316.                                 c.ssh_control_socket_name="/tmp/sshnfs_ssh-control-socket_logname${LOGNAME}_ppid${PPID}_pid$$"
  317.  
  318.                                 # Forward NFS port from server to local machine
  319.                                 # Notes:
  320.                                 # - We use $ ssh -M ... # here as a way to terminate the port
  321.                                 # forwarding process later using "-O exit" without the need
  322.                                 # for a pid
  323.                                 print -u2 -f $"# Please enter the login data for NFS server (%s):\n" \
  324.                                         "${c.nfsserver_ssh_login_name}@${c.nfs_server.host}"
  325.                                 ssh \
  326.                                         -L "${c.local_forward_port}:localhost:${c.nfs_server.port}" \
  327.                                         -M -S "${c.ssh_control_socket_name}" \
  328.                                         -N \
  329.                                         -f -o 'ExitOnForwardFailure=yes' \
  330.                                         "${c.ssh_jumphost_args[@]}" \
  331.                                         "${c.nfsserver_ssh_login_name}@${c.nfs_server.host}"
  332.                                 if (( $? != 0 )) ; then
  333.                                         print -u2 -f $"%s: NFS forwarding ssh failed with error code %d\n" "$0" $?
  334.                                         return 1
  335.                                 fi
  336.  
  337.                                 # debug
  338.                                 ${mydebug} && ssh -S "${c.ssh_control_socket_name}" -O 'check' "root@${c.nfs_server.host}"
  339.  
  340.                                 print -u2 -f $"# Use this to mount the directory:\n"
  341.                                 print -u2 -f $"# $ mkdir /mnt_nfs\n"
  342.                                 print -u2 -f $"# $ mount -vvv -t nfs -o vers=4,port=%d localhost:/%s /mnt_nfs\n" \
  343.                                         c.destination_nfs_port \
  344.                                         "${c.nfs_server.path}"
  345.  
  346.                                 # add NFS forwarding options to main ssh argument list
  347.                                 c.args=(
  348.                                         '-R' "${c.destination_nfs_port}:localhost:${c.local_forward_port}"
  349.                                         '-o' 'ExitOnForwardFailure=yes'
  350.                                         "${c.args[@]}"
  351.                                 )
  352.                                 ;;
  353.                         'nfs')
  354.                                 if [[ -v c.ssh_jumphost_args ]] ; then
  355.                                         print -u2 -f $"$s: Error: SSHNFSJumphost cannot be used for nfs://-URLs\n" "$0"
  356.                                         return 2
  357.                                 fi
  358.  
  359.                                 integer myuid=$(id -u)
  360.                                 integer mypid=$$ # used to circumvent ksh93 -n warning
  361.  
  362.                                 # TCP port on destination machine where we forward the
  363.                                 # NFS port from the server
  364.                                 integer c.destination_nfs_port=33049
  365.  
  366.                                 # try to adjust c.destination_nfs_port so that multiple sshnfs.ksh
  367.                                 # sessions do intefere with each other
  368.                                 # (16381 is a prime number)
  369.                                 (( c.destination_nfs_port += (mypid+myuid+PPID) % 16381 ))
  370.  
  371.                                 print -u2 -f $"# Use this to mount the directory:\n"
  372.                                 print -u2 -f $"# $ mkdir /mnt_nfs\n"
  373.                                 print -u2 -f $"# $ mount -vvv -t nfs -o vers=4,port=%d localhost:/%s /mnt_nfs\n" \
  374.                                         c.destination_nfs_port \
  375.                                         "${c.nfs_server.path}"
  376.  
  377.                                 # add NFS forwarding options to main ssh argument list
  378.                                 c.args=(
  379.                                         '-R' "${c.destination_nfs_port}:${c.nfs_server.host}:${c.nfs_server.port}"
  380.                                         '-o' 'ExitOnForwardFailure=yes'
  381.                                         "${c.args[@]}"
  382.                                 )
  383.                                 ;;
  384.                         *)
  385.                                 print -u2 -f $"%s: Unknown protocol %q\n" "$0" "${c.nfs_server.protocol}"
  386.                                 return 2
  387.                                 ;;
  388.                 esac
  389.         fi
  390.  
  391.         # debug: print application data (compound c)
  392.         ${mydebug} && print -v c
  393.  
  394.         print -u2 -f $"# ssh login data for destination machine:\n"
  395.         ssh "${c.args[@]}" ; (( retval=$? ))
  396.  
  397.         if [[ -v c.ssh_control_socket_name ]] ; then
  398.                 ssh -S "${c.ssh_control_socket_name}" -O 'exit' "root@${c.nfs_server.host}"
  399.         fi
  400.  
  401.         wait
  402.  
  403.         return $retval
  404. }
  405.  
  406. main "$@"
  407. exit $?
  408.  
  409. # EOF.

Submit a correction or amendment below (click here to make a fresh posting)
After submitting an amendment, you'll be able to view the differences between the old and new posts easily.

Syntax highlighting:

To highlight particular lines, prefix each line with {%HIGHLIGHT}




All content is user-submitted.
The administrators of this site (kpaste.net) are not responsible for their content.
Abuse reports should be emailed to us at