// Copyright 2017 gf Author(https://gitee.com/johng/gf). All Rights Reserved. // // This Source Code Form is subject to the terms of the MIT License. // If a copy of the MIT was not distributed with this file, // You can obtain one at https://gitee.com/johng/gf. // 请求处理. package ghttp import ( "os" "fmt" "sort" "reflect" "strings" "net/url" "net/http" "path/filepath" "gitee.com/johng/gf/g/os/gfile" "gitee.com/johng/gf/g/encoding/ghtml" ) // 默认HTTP Server处理入口,http包底层默认使用了gorutine异步处理请求,所以这里不再异步执行 func (s *Server)defaultHttpHandle(w http.ResponseWriter, r *http.Request) { s.handleRequest(w, r) } // 执行处理HTTP请求 // 首先,查找是否有对应域名的处理接口配置; // 其次,如果没有对应的自定义处理接口配置,那么走默认的域名处理接口配置; // 最后,如果以上都没有找到处理接口,那么进行文件处理; func (s *Server)handleRequest(w http.ResponseWriter, r *http.Request) { // 去掉末尾的"/"号 if r.URL.Path != "/" { r.URL.Path = strings.TrimRight(r.URL.Path, "/") } // 创建请求处理对象 request := newRequest(s, r, w) // 事件 - BeforeServe s.callHookHandler(request, "BeforeServe") if h := s.getHandler(request); h != nil { s.callHandler(h, request) } else { s.serveFile(request) } // 事件 - AfterServe s.callHookHandler(request, "AfterServe") // 事件 - BeforeOutput s.callHookHandler(request, "BeforeOutput") // 输出Cookie request.Cookie.Output() // 输出缓冲区 request.Response.OutputBuffer() // 事件 - AfterOutput s.callHookHandler(request, "AfterOutput") // 将Request对象指针丢到队列中异步处理 s.closeQueue.PushBack(request) } // 初始化控制器 func (s *Server)callHandler(h *HandlerItem, r *Request) { if h.faddr == nil { // 新建一个控制器对象处理请求 c := reflect.New(h.ctype) c.MethodByName("Init").Call([]reflect.Value{reflect.ValueOf(r)}) if !r.IsExited() { c.MethodByName(h.fname).Call(nil) } c.MethodByName("Shut").Call([]reflect.Value{reflect.ValueOf(r)}) } else { if !r.IsExited() { h.faddr(r) } } } // 处理静态文件请求 func (s *Server)serveFile(r *Request) { uri := r.URL.String() if s.config.ServerRoot != "" { // 获取文件的绝对路径 path := strings.TrimRight(s.config.ServerRoot, string(filepath.Separator)) path = path + uri path = gfile.RealPath(path) if path != "" { s.doServeFile(r, path) } else { r.Response.WriteStatus(http.StatusNotFound) } } else { r.Response.WriteStatus(http.StatusNotFound) } } // http server静态文件处理 func (s *Server)doServeFile(r *Request, path string) { f, err := os.Open(path) if err != nil { return } info, _ := f.Stat() if info.IsDir() { if len(s.config.IndexFiles) > 0 { for _, file := range s.config.IndexFiles { fpath := path + "/" + file if gfile.Exists(fpath) { f.Close() s.doServeFile(r, fpath) return } } } if s.config.IndexFolder { s.listDir(r, f) } else { r.Response.WriteStatus(http.StatusForbidden) } } else { // 读取文件内容返回 http.ServeContent(r.Response.ResponseWriter, &r.Request, info.Name(), info.ModTime(), f) } f.Close() } // 目录列表 func (s *Server)listDir(r *Request, f http.File) { dirs, err := f.Readdir(-1) if err != nil { r.Response.WriteStatus(http.StatusInternalServerError, "Error reading directory") return } sort.Slice(dirs, func(i, j int) bool { return dirs[i].Name() < dirs[j].Name() }) r.Response.Header().Set("Content-Type", "text/html; charset=utf-8") r.Response.Write("
\n")
    for _, d := range dirs {
        name := d.Name()
        if d.IsDir() {
            name += "/"
        }
        u := url.URL{Path: name}
        r.Response.Write(fmt.Sprintf("%s\n", u.String(), ghtml.SpecialChars(name)))
    }
    r.Response.Write("
\n") }