You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

320 lines
8.5 KiB

  1. // Copyright 2011 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package ssh_test
  5. import (
  6. "bufio"
  7. "bytes"
  8. "fmt"
  9. "io/ioutil"
  10. "log"
  11. "net"
  12. "net/http"
  13. "os"
  14. "path/filepath"
  15. "strings"
  16. "golang.org/x/crypto/ssh"
  17. "golang.org/x/crypto/ssh/terminal"
  18. )
  19. func ExampleNewServerConn() {
  20. // Public key authentication is done by comparing
  21. // the public key of a received connection
  22. // with the entries in the authorized_keys file.
  23. authorizedKeysBytes, err := ioutil.ReadFile("authorized_keys")
  24. if err != nil {
  25. log.Fatalf("Failed to load authorized_keys, err: %v", err)
  26. }
  27. authorizedKeysMap := map[string]bool{}
  28. for len(authorizedKeysBytes) > 0 {
  29. pubKey, _, _, rest, err := ssh.ParseAuthorizedKey(authorizedKeysBytes)
  30. if err != nil {
  31. log.Fatal(err)
  32. }
  33. authorizedKeysMap[string(pubKey.Marshal())] = true
  34. authorizedKeysBytes = rest
  35. }
  36. // An SSH server is represented by a ServerConfig, which holds
  37. // certificate details and handles authentication of ServerConns.
  38. config := &ssh.ServerConfig{
  39. // Remove to disable password auth.
  40. PasswordCallback: func(c ssh.ConnMetadata, pass []byte) (*ssh.Permissions, error) {
  41. // Should use constant-time compare (or better, salt+hash) in
  42. // a production setting.
  43. if c.User() == "testuser" && string(pass) == "tiger" {
  44. return nil, nil
  45. }
  46. return nil, fmt.Errorf("password rejected for %q", c.User())
  47. },
  48. // Remove to disable public key auth.
  49. PublicKeyCallback: func(c ssh.ConnMetadata, pubKey ssh.PublicKey) (*ssh.Permissions, error) {
  50. if authorizedKeysMap[string(pubKey.Marshal())] {
  51. return &ssh.Permissions{
  52. // Record the public key used for authentication.
  53. Extensions: map[string]string{
  54. "pubkey-fp": ssh.FingerprintSHA256(pubKey),
  55. },
  56. }, nil
  57. }
  58. return nil, fmt.Errorf("unknown public key for %q", c.User())
  59. },
  60. }
  61. privateBytes, err := ioutil.ReadFile("id_rsa")
  62. if err != nil {
  63. log.Fatal("Failed to load private key: ", err)
  64. }
  65. private, err := ssh.ParsePrivateKey(privateBytes)
  66. if err != nil {
  67. log.Fatal("Failed to parse private key: ", err)
  68. }
  69. config.AddHostKey(private)
  70. // Once a ServerConfig has been configured, connections can be
  71. // accepted.
  72. listener, err := net.Listen("tcp", "0.0.0.0:2022")
  73. if err != nil {
  74. log.Fatal("failed to listen for connection: ", err)
  75. }
  76. nConn, err := listener.Accept()
  77. if err != nil {
  78. log.Fatal("failed to accept incoming connection: ", err)
  79. }
  80. // Before use, a handshake must be performed on the incoming
  81. // net.Conn.
  82. conn, chans, reqs, err := ssh.NewServerConn(nConn, config)
  83. if err != nil {
  84. log.Fatal("failed to handshake: ", err)
  85. }
  86. log.Printf("logged in with key %s", conn.Permissions.Extensions["pubkey-fp"])
  87. // The incoming Request channel must be serviced.
  88. go ssh.DiscardRequests(reqs)
  89. // Service the incoming Channel channel.
  90. for newChannel := range chans {
  91. // Channels have a type, depending on the application level
  92. // protocol intended. In the case of a shell, the type is
  93. // "session" and ServerShell may be used to present a simple
  94. // terminal interface.
  95. if newChannel.ChannelType() != "session" {
  96. newChannel.Reject(ssh.UnknownChannelType, "unknown channel type")
  97. continue
  98. }
  99. channel, requests, err := newChannel.Accept()
  100. if err != nil {
  101. log.Fatalf("Could not accept channel: %v", err)
  102. }
  103. // Sessions have out-of-band requests such as "shell",
  104. // "pty-req" and "env". Here we handle only the
  105. // "shell" request.
  106. go func(in <-chan *ssh.Request) {
  107. for req := range in {
  108. req.Reply(req.Type == "shell", nil)
  109. }
  110. }(requests)
  111. term := terminal.NewTerminal(channel, "> ")
  112. go func() {
  113. defer channel.Close()
  114. for {
  115. line, err := term.ReadLine()
  116. if err != nil {
  117. break
  118. }
  119. fmt.Println(line)
  120. }
  121. }()
  122. }
  123. }
  124. func ExampleHostKeyCheck() {
  125. // Every client must provide a host key check. Here is a
  126. // simple-minded parse of OpenSSH's known_hosts file
  127. host := "hostname"
  128. file, err := os.Open(filepath.Join(os.Getenv("HOME"), ".ssh", "known_hosts"))
  129. if err != nil {
  130. log.Fatal(err)
  131. }
  132. defer file.Close()
  133. scanner := bufio.NewScanner(file)
  134. var hostKey ssh.PublicKey
  135. for scanner.Scan() {
  136. fields := strings.Split(scanner.Text(), " ")
  137. if len(fields) != 3 {
  138. continue
  139. }
  140. if strings.Contains(fields[0], host) {
  141. var err error
  142. hostKey, _, _, _, err = ssh.ParseAuthorizedKey(scanner.Bytes())
  143. if err != nil {
  144. log.Fatalf("error parsing %q: %v", fields[2], err)
  145. }
  146. break
  147. }
  148. }
  149. if hostKey == nil {
  150. log.Fatalf("no hostkey for %s", host)
  151. }
  152. config := ssh.ClientConfig{
  153. User: os.Getenv("USER"),
  154. HostKeyCallback: ssh.FixedHostKey(hostKey),
  155. }
  156. _, err = ssh.Dial("tcp", host+":22", &config)
  157. log.Println(err)
  158. }
  159. func ExampleDial() {
  160. var hostKey ssh.PublicKey
  161. // An SSH client is represented with a ClientConn.
  162. //
  163. // To authenticate with the remote server you must pass at least one
  164. // implementation of AuthMethod via the Auth field in ClientConfig,
  165. // and provide a HostKeyCallback.
  166. config := &ssh.ClientConfig{
  167. User: "username",
  168. Auth: []ssh.AuthMethod{
  169. ssh.Password("yourpassword"),
  170. },
  171. HostKeyCallback: ssh.FixedHostKey(hostKey),
  172. }
  173. client, err := ssh.Dial("tcp", "yourserver.com:22", config)
  174. if err != nil {
  175. log.Fatal("Failed to dial: ", err)
  176. }
  177. // Each ClientConn can support multiple interactive sessions,
  178. // represented by a Session.
  179. session, err := client.NewSession()
  180. if err != nil {
  181. log.Fatal("Failed to create session: ", err)
  182. }
  183. defer session.Close()
  184. // Once a Session is created, you can execute a single command on
  185. // the remote side using the Run method.
  186. var b bytes.Buffer
  187. session.Stdout = &b
  188. if err := session.Run("/usr/bin/whoami"); err != nil {
  189. log.Fatal("Failed to run: " + err.Error())
  190. }
  191. fmt.Println(b.String())
  192. }
  193. func ExamplePublicKeys() {
  194. var hostKey ssh.PublicKey
  195. // A public key may be used to authenticate against the remote
  196. // server by using an unencrypted PEM-encoded private key file.
  197. //
  198. // If you have an encrypted private key, the crypto/x509 package
  199. // can be used to decrypt it.
  200. key, err := ioutil.ReadFile("/home/user/.ssh/id_rsa")
  201. if err != nil {
  202. log.Fatalf("unable to read private key: %v", err)
  203. }
  204. // Create the Signer for this private key.
  205. signer, err := ssh.ParsePrivateKey(key)
  206. if err != nil {
  207. log.Fatalf("unable to parse private key: %v", err)
  208. }
  209. config := &ssh.ClientConfig{
  210. User: "user",
  211. Auth: []ssh.AuthMethod{
  212. // Use the PublicKeys method for remote authentication.
  213. ssh.PublicKeys(signer),
  214. },
  215. HostKeyCallback: ssh.FixedHostKey(hostKey),
  216. }
  217. // Connect to the remote server and perform the SSH handshake.
  218. client, err := ssh.Dial("tcp", "host.com:22", config)
  219. if err != nil {
  220. log.Fatalf("unable to connect: %v", err)
  221. }
  222. defer client.Close()
  223. }
  224. func ExampleClient_Listen() {
  225. var hostKey ssh.PublicKey
  226. config := &ssh.ClientConfig{
  227. User: "username",
  228. Auth: []ssh.AuthMethod{
  229. ssh.Password("password"),
  230. },
  231. HostKeyCallback: ssh.FixedHostKey(hostKey),
  232. }
  233. // Dial your ssh server.
  234. conn, err := ssh.Dial("tcp", "localhost:22", config)
  235. if err != nil {
  236. log.Fatal("unable to connect: ", err)
  237. }
  238. defer conn.Close()
  239. // Request the remote side to open port 8080 on all interfaces.
  240. l, err := conn.Listen("tcp", "0.0.0.0:8080")
  241. if err != nil {
  242. log.Fatal("unable to register tcp forward: ", err)
  243. }
  244. defer l.Close()
  245. // Serve HTTP with your SSH server acting as a reverse proxy.
  246. http.Serve(l, http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
  247. fmt.Fprintf(resp, "Hello world!\n")
  248. }))
  249. }
  250. func ExampleSession_RequestPty() {
  251. var hostKey ssh.PublicKey
  252. // Create client config
  253. config := &ssh.ClientConfig{
  254. User: "username",
  255. Auth: []ssh.AuthMethod{
  256. ssh.Password("password"),
  257. },
  258. HostKeyCallback: ssh.FixedHostKey(hostKey),
  259. }
  260. // Connect to ssh server
  261. conn, err := ssh.Dial("tcp", "localhost:22", config)
  262. if err != nil {
  263. log.Fatal("unable to connect: ", err)
  264. }
  265. defer conn.Close()
  266. // Create a session
  267. session, err := conn.NewSession()
  268. if err != nil {
  269. log.Fatal("unable to create session: ", err)
  270. }
  271. defer session.Close()
  272. // Set up terminal modes
  273. modes := ssh.TerminalModes{
  274. ssh.ECHO: 0, // disable echoing
  275. ssh.TTY_OP_ISPEED: 14400, // input speed = 14.4kbaud
  276. ssh.TTY_OP_OSPEED: 14400, // output speed = 14.4kbaud
  277. }
  278. // Request pseudo terminal
  279. if err := session.RequestPty("xterm", 40, 80, modes); err != nil {
  280. log.Fatal("request for pseudo terminal failed: ", err)
  281. }
  282. // Start remote shell
  283. if err := session.Shell(); err != nil {
  284. log.Fatal("failed to start shell: ", err)
  285. }
  286. }