go - 返回响应后关闭 HTTP 服务器

标签 go httpserver

我正在构建一个基于命令行的小型 Go 机器人,它可以与 Instagram API 进行交互。

Instagram API 是基于 OAuth 的,因此对于基于命令行的应用来说并不是特别好。

为了解决这个问题,我在浏览器中打开适当的授权 URL 并使用我为重定向 URI 启动的本地服务器 - 这样我就可以捕获并优雅地显示访问 token ,而不是用户需要获取手动从 URL 获取。

到目前为止一切顺利,应用程序可以成功打开浏览器访问授权 URL,您授权它,它会将您重定向到本地 HTTP 服务器。

现在,在向用户显示访问 token 后我不需要 HTTP 服务器,因此我想在执行此操作后手动关闭服务器。

为此,我从这个 answer 中得到灵感并鼓起以下内容:

package main

import (
    "fmt"
    "io"
    "log"
    "net/http"
    "os/exec"
    "runtime"
    "time"
)

var client_id = "my_client_id"
var client_secret = "my_client_secret"
var redirect_url = "http://localhost:8000/instagram/callback"

func main() {

    srv := startHttpServer()

    openbrowser(fmt.Sprintf("https://api.instagram.com/oauth/authorize/?client_id=%v&redirect_uri=%v&response_type=code", client_id, redirect_url))

    // Backup to gracefully shutdown the server
    time.Sleep(20 * time.Second)
    if err := srv.Shutdown(nil); err != nil {
        panic(err) // failure/timeout shutting down the server gracefully
    }
}

func showTokenToUser(w http.ResponseWriter, r *http.Request, srv *http.Server) {
    io.WriteString(w, fmt.Sprintf("Your access token is: %v", r.URL.Query().Get("code")))
    if err := srv.Shutdown(nil); err != nil {
        log.Fatal(err) // failure/timeout shutting down the server gracefully
    }
}

func startHttpServer() *http.Server {
    srv := &http.Server{Addr: ":8000"}

    http.HandleFunc("/instagram/callback", func(w http.ResponseWriter, r *http.Request) {
        showTokenToUser(w, r, srv)
    })

    go func() {
        if err := srv.ListenAndServe(); err != nil {
            // cannot panic, because this probably is an intentional close
            log.Printf("Httpserver: ListenAndServe() error: %s", err)
        }
    }()

    // returning reference so caller can call Shutdown()
    return srv
}

func openbrowser(url string) {
    var err error

    switch runtime.GOOS {
    case "linux":
        err = exec.Command("xdg-open", url).Start()
    case "windows":
        err = exec.Command("rundll32", "url.dll,FileProtocolHandler", url).Start()
    case "darwin":
        err = exec.Command("open", url).Start()
    default:
        err = fmt.Errorf("unsupported platform")
    }
    if err != nil {
        log.Fatal(err)
    }
}

但是,以上会导致这个错误:

2017/11/23 16:02:03 Httpserver: ListenAndServe() error: http: Server closed

2017/11/23 16:02:03 http: panic serving [::1]:61793: runtime error: invalid memory address or nil pointer dereference

如果我在处理程序中注释掉这些行,那么它就可以完美地工作,尽管在我点击回调路由时没有关闭服务器:

if err := srv.Shutdown(nil); err != nil {
    log.Fatal(err) // failure/timeout shutting down the server gracefully
}

我哪里出错了?我需要更改什么才能在向用户显示文本后点击回调路由时关闭服务器。

最佳答案

  1. 您可以使用 context.WithCancel:
package main

import (
    "context"
    "io"
    "log"
    "net/http"
)

func main() {
    ctx, cancel := context.WithCancel(context.Background())

    http.HandleFunc("/quit", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Bye\n")
        cancel()
    })
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Hi\n")
    })

    srv := &http.Server{Addr: ":8080"}
    go func() {
        err := srv.ListenAndServe()
        if err != http.ErrServerClosed {
            log.Println(err)
        }
    }()

    <-ctx.Done() // wait for the signal to gracefully shutdown the server

    // gracefully shutdown the server:
    // waiting indefinitely for connections to return to idle and then shut down.
    err := srv.Shutdown(context.Background())
    if err != nil {
        log.Println(err)
    }

    log.Println("done.")
}

  1. The same Context可以传递给在不同 goroutine 中运行的函数:

"Contexts are safe for simultaneous use by multiple goroutines."

您可以使用相同的 context - 如果您不想等待 extera:

package main

import (
    "context"
    "io"
    "log"
    "net/http"
)

func main() {
    ctx, cancel := context.WithCancel(context.Background())
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Hi\n")
    })
    http.HandleFunc("/quit", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Bye\n")
        cancel()
    })
    srv := &http.Server{Addr: ":8080"}
    go func() {
        if err := srv.ListenAndServe(); err != nil {
            log.Printf("Httpserver: ListenAndServe() error: %s", err)
        }
    }()
    <-ctx.Done()
    // if err := srv.Shutdown(ctx); err != nil && err != context.Canceled {
    //  log.Println(err)
    // }
    log.Println("done.")
}

Server.Shutdown :
Shutdown 在不中断任何事件连接的情况下优雅地关闭服务器。 Shutdown 的工作原理是首先关闭所有打开的监听器,然后关闭所有空闲连接,然后无限期等待连接返回空闲状态,然后关闭。如果提供的上下文在关闭完成之前过期,则 Shutdown 返回上下文的错误,否则它返回关闭服务器的底层监听器返回的任何错误。

当调用 Shutdown 时,Serve、ListenAndServe 和 ListenAndServeTLS 立即返回 ErrServerClosed。确保程序不会退出,而是等待 Shutdown 返回。

Shutdown 不会尝试关闭或等待被劫持的连接,例如 WebSockets。如果需要,Shutdown 的调用者应单独通知此类长期连接关闭并等待它们关闭。有关注册关机通知功能的方法,请参阅 RegisterOnShutdown。

一旦在服务器上调用了 Shutdown,就不能再使用它;以后调用 Serve 等方法将返回 ErrServerClosed。

关于go - 返回响应后关闭 HTTP 服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47448666/

相关文章:

go - Gin /golang - 空请求体

go - 如何在本地处理多个 Go 项目?

高语 : Reading a File Basics

Java httpServer 不同请求方式的基本认证

go - 如何迭代 go time.Tick channel ?

go - Go中的Os Exec Sudo命令

java - Apache http 服务器重写 url 失败

java httpserver处理POST请求并读取html表单信息

PowerShell 脚本返回意外输出(随机数)