// AttachFile method attaches a file to an alert at OpsGenie. func (cli *OpsGenieAlertClient) AttachFile(req alerts.AttachFileAlertRequest) (*alerts.AttachFileAlertResponse, error) { req.APIKey = cli.apiKey var b bytes.Buffer w := multipart.NewWriter(&b) path := req.Attachment.Name() file, err := os.Open(path) defer file.Close() if err != nil { message := "Attachment can not be opened for reading. " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } // add the attachment fw, err := w.CreateFormFile("attachment", filepath.Base(path)) if err != nil { message := "Can not build the request with the field attachment. " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } if _, err := io.Copy(fw, file); err != nil { message := "Can not copy the attachment into the request. " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } // Add the other fields // empty fields should not be placed into the request // otherwise it yields an incomplete boundary exception if req.APIKey != "" { if err = writeField(*w, "apiKey", req.APIKey); err != nil { return nil, err } } if req.ID != "" { if err = writeField(*w, "id", req.ID); err != nil { return nil, err } } if req.Alias != "" { if err = writeField(*w, "alias", req.Alias); err != nil { return nil, err } } if req.User != "" { if err = writeField(*w, "user", req.User); err != nil { return nil, err } } if req.Source != "" { if err = writeField(*w, "source", req.Source); err != nil { return nil, err } } if req.IndexFile != "" { if err = writeField(*w, "indexFile", req.IndexFile); err != nil { return nil, err } } if req.Note != "" { if err = writeField(*w, "note", req.Note); err != nil { return nil, err } } w.Close() httpReq, err := http.NewRequest("POST", cli.opsGenieAPIURL+attachFileAlertURL, &b) if err != nil { message := "Can not create the multipart/form-data request. " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } httpReq.Header.Set("Content-Type", w.FormDataContentType()) transport := &http.Transport{ TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, Proxy: http.ProxyFromEnvironment, Dial: func(netw, addr string) (net.Conn, error) { conn, err := net.DialTimeout(netw, addr, cli.httpTransportSettings.ConnectionTimeout) if err != nil { message := "Error occurred while connecting: " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } conn.SetDeadline(time.Now().Add(cli.httpTransportSettings.RequestTimeout)) return conn, nil }, } client := &http.Client{Transport: transport} // proxy settings if cli.proxy != nil { proxyURL, proxyErr := url.Parse(cli.proxy.toString()) if proxyErr != nil { message := "Can not set the proxy configuration " + proxyErr.Error() logging.Logger().Warn(message) return nil, errors.New(message) } transport.Proxy = http.ProxyURL(proxyURL) } url := httpReq.URL.String() logging.Logger().Info("Executing OpsGenie request to [" + url + "] with multipart data.") var res *http.Response for i := 0; i < cli.httpTransportSettings.MaxRetryAttempts; i++ { res, err = client.Do(httpReq) if err == nil { defer res.Body.Close() break } if res != nil { logging.Logger().Info(fmt.Sprintf("Retrying request [%s] ResponseCode:[%d]. RetryCount: %d", url, res.StatusCode, (i + 1))) } else { logging.Logger().Info(fmt.Sprintf("Retrying request [%s] Reason:[%s]. RetryCount: %d", url, err.Error(), (i + 1))) } time.Sleep(timeSleepBetweenRequests) } if err != nil { message := "Can not attach the file, unable to send the request. " + err.Error() logging.Logger().Warn(message) return nil, errors.New(message) } httpStatusCode := res.StatusCode if httpStatusCode >= 400 { body, err := ioutil.ReadAll(res.Body) if err == nil { return nil, errorMessage(httpStatusCode, string(body[:])) } message := fmt.Sprint("Couldn't read the response, %s", err.Error()) logging.Logger().Warn(message) return nil, errors.New(message) } attachFileAlertResp := alerts.AttachFileAlertResponse{Status: res.Status, Code: res.StatusCode} return &attachFileAlertResp, nil }