mirror of https://github.com/minio/minio.git
				
				
				
			
		
			
				
	
	
		
			119 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			119 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			Go
		
	
	
	
| /*
 | |
|  * Minimalist Object Storage, (C) 2015 Minio, Inc.
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| package main
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"os"
 | |
| 	"time"
 | |
| )
 | |
| 
 | |
| type accessLogHandler struct {
 | |
| 	http.Handler
 | |
| 	accessLogFile *os.File
 | |
| }
 | |
| 
 | |
| // LogMessage is a serializable json log message
 | |
| type LogMessage struct {
 | |
| 	StartTime     time.Time
 | |
| 	Duration      time.Duration
 | |
| 	StatusMessage string // human readable http status message
 | |
| 	ContentLength string // human readable content length
 | |
| 
 | |
| 	// HTTP detailed message
 | |
| 	HTTP struct {
 | |
| 		ResponseHeaders http.Header
 | |
| 		Request         struct {
 | |
| 			Method     string
 | |
| 			URL        *url.URL
 | |
| 			Proto      string // "HTTP/1.0"
 | |
| 			ProtoMajor int    // 1
 | |
| 			ProtoMinor int    // 0
 | |
| 			Header     http.Header
 | |
| 			Host       string
 | |
| 			Form       url.Values
 | |
| 			PostForm   url.Values
 | |
| 			Trailer    http.Header
 | |
| 			RemoteAddr string
 | |
| 			RequestURI string
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (h *accessLogHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
 | |
| 	message, err := getLogMessage(w, req)
 | |
| 	fatalIf(err, "Unable to parse HTTP request and response fields.")
 | |
| 	_, err = h.accessLogFile.Write(message)
 | |
| 	fatalIf(err, "Unable to log HTTP access.")
 | |
| 
 | |
| 	h.Handler.ServeHTTP(w, req)
 | |
| }
 | |
| 
 | |
| func getLogMessage(w http.ResponseWriter, req *http.Request) ([]byte, error) {
 | |
| 	logMessage := &LogMessage{
 | |
| 		StartTime: time.Now().UTC(),
 | |
| 	}
 | |
| 	// store lower level details
 | |
| 	logMessage.HTTP.ResponseHeaders = w.Header()
 | |
| 	logMessage.HTTP.Request = struct {
 | |
| 		Method     string
 | |
| 		URL        *url.URL
 | |
| 		Proto      string // "HTTP/1.0"
 | |
| 		ProtoMajor int    // 1
 | |
| 		ProtoMinor int    // 0
 | |
| 		Header     http.Header
 | |
| 		Host       string
 | |
| 		Form       url.Values
 | |
| 		PostForm   url.Values
 | |
| 		Trailer    http.Header
 | |
| 		RemoteAddr string
 | |
| 		RequestURI string
 | |
| 	}{
 | |
| 		Method:     req.Method,
 | |
| 		URL:        req.URL,
 | |
| 		Proto:      req.Proto,
 | |
| 		ProtoMajor: req.ProtoMajor,
 | |
| 		ProtoMinor: req.ProtoMinor,
 | |
| 		Header:     req.Header,
 | |
| 		Host:       req.Host,
 | |
| 		Form:       req.Form,
 | |
| 		PostForm:   req.PostForm,
 | |
| 		Trailer:    req.Header,
 | |
| 		RemoteAddr: req.RemoteAddr,
 | |
| 		RequestURI: req.RequestURI,
 | |
| 	}
 | |
| 
 | |
| 	// logMessage.HTTP.Request = req
 | |
| 	logMessage.Duration = time.Now().UTC().Sub(logMessage.StartTime)
 | |
| 	js, err := json.Marshal(logMessage)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	js = append(js, byte('\n')) // append a new line
 | |
| 	return js, nil
 | |
| }
 | |
| 
 | |
| // setAccessLogHandler logs requests
 | |
| func setAccessLogHandler(h http.Handler) http.Handler {
 | |
| 	file, err := os.OpenFile("access.log", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0600)
 | |
| 	fatalIf(err, "Failed top open access log.")
 | |
| 
 | |
| 	return &accessLogHandler{Handler: h, accessLogFile: file}
 | |
| }
 |