mirror of
https://github.com/mattermost/mattermost.git
synced 2026-05-28 04:35:04 -04:00
Some checks are pending
API / build (push) Waiting to run
Server CI / Compute Go Version (push) Waiting to run
Server CI / Check mocks (push) Blocked by required conditions
Server CI / Check go mod tidy (push) Blocked by required conditions
Server CI / Check go fix (push) Blocked by required conditions
Server CI / check-style (push) Blocked by required conditions
Server CI / Check serialization methods for hot structs (push) Blocked by required conditions
Server CI / Vet API (push) Blocked by required conditions
Server CI / Check migration files (push) Blocked by required conditions
Server CI / Generate email templates (push) Blocked by required conditions
Server CI / Check store layers (push) Blocked by required conditions
Server CI / Check mmctl docs (push) Blocked by required conditions
Server CI / Postgres (shard 0) (push) Blocked by required conditions
Server CI / Postgres (shard 1) (push) Blocked by required conditions
Server CI / Postgres (shard 2) (push) Blocked by required conditions
Server CI / Postgres (shard 3) (push) Blocked by required conditions
Server CI / Merge Postgres Test Results (push) Blocked by required conditions
Server CI / Elasticsearch v8 Compatibility (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 0) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 1) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 2) (push) Blocked by required conditions
Server CI / Postgres FIPS (shard 3) (push) Blocked by required conditions
Server CI / Merge Postgres FIPS Test Results (push) Blocked by required conditions
Server CI / Run mmctl tests (push) Blocked by required conditions
Server CI / Run mmctl tests (FIPS) (push) Blocked by required conditions
Server CI / Build mattermost server app (push) Blocked by required conditions
Tools CI / check-style (mattermost-govet) (push) Waiting to run
Tools CI / Test (mattermost-govet) (push) Waiting to run
Web App CI / check-lint (push) Waiting to run
Web App CI / check-i18n (push) Blocked by required conditions
Web App CI / check-external-links (push) Blocked by required conditions
Web App CI / check-types (push) Blocked by required conditions
Web App CI / test (platform) (push) Blocked by required conditions
Web App CI / test (mattermost-redux) (push) Blocked by required conditions
Web App CI / test (channels shard 1/4) (push) Blocked by required conditions
Web App CI / test (channels shard 2/4) (push) Blocked by required conditions
Web App CI / test (channels shard 3/4) (push) Blocked by required conditions
Web App CI / test (channels shard 4/4) (push) Blocked by required conditions
Web App CI / upload-coverage (push) Blocked by required conditions
Web App CI / build (push) Blocked by required conditions
YAML Lint / yamllint (push) Waiting to run
* Simplify invite_people email parsing Replace backwards in-place mutation loop with a straightforward forward filter into a new slice. Extract into parseEmailList so the logic can be unit tested directly. * MM-68150: Upgrade golangci-lint to v2.12.2 Remove //go:fix inline from NewPointer, which is a generic function not yet supported by the inline analyzer, and fix 11 slicesbackward modernize issues flagged by the new version. * MM-68150: Enable all linters by default; disable those with >20 existing issues Switch from opt-in (default: none) to opt-out (default: all) so new linters added to golangci-lint are evaluated automatically. Explicitly disable every linter that has more than 20 pre-existing violations, deferring those for later cleanup. Also disable a handful of linters whose violations are intentional patterns in this codebase (nilerr, dogsled, sqlclosecheck, iotamixing, predeclared, containedctx, iface, gocheckcompilerdirectives, promlinter, goprintffuncname, gomoddirectives). * MM-68150: Fix mirror linter issues Replace Write([]byte(s)) with WriteString(s), and FindIndex([]byte(s)) with FindStringIndex(s), to avoid unnecessary allocations. * MM-68150: Fix nosprintfhostport linter issue Use net.JoinHostPort to construct host:port strings instead of fmt.Sprintf with a manually formatted pattern. * MM-68150: Fix rowserrcheck and sqlclosecheck linter issues Check rows.Err() after iteration loops in schema_dump.go. In the sqlx_wrapper test, defer rows.Close() rather than closing inline. * MM-68150: Fix nilnesserr linter issues — wrong variable in error handlers In 11 places, a stale variable (often the outer err from a prior assignment) was used instead of the freshly-checked error variable (appErr, rowErr, jsonErr, writeErr, esErr). Each produces a typed-nil wrapped in a non-nil interface, silently discarding the real error. * MM-68150: Add i18n string for app.compile_csv_chunks.write_error --------- Co-authored-by: Mattermost Build <build@mattermost.com>
183 lines
3.9 KiB
Go
183 lines
3.9 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package mail
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
InbucketAPI = "/api/v1/mailbox/"
|
|
)
|
|
|
|
// OutputJSONHeader holds the received Header to test sending emails (inbucket)
|
|
type JSONMessageHeaderInbucket []struct {
|
|
Mailbox string
|
|
ID string `json:"Id"`
|
|
From, Subject, Date string
|
|
To []string
|
|
Size int
|
|
}
|
|
|
|
// OutputJSONMessage holds the received Message fto test sending emails (inbucket)
|
|
type JSONMessageInbucket struct {
|
|
Mailbox string
|
|
ID string `json:"Id"`
|
|
From, Subject, Date string
|
|
Size int
|
|
Header map[string][]string
|
|
Body struct {
|
|
Text string
|
|
HTML string `json:"Html"`
|
|
}
|
|
Attachments []struct {
|
|
Filename string
|
|
ContentType string `json:"content-type"`
|
|
DownloadLink string `json:"download-link"`
|
|
Bytes []byte `json:"-"`
|
|
}
|
|
}
|
|
|
|
func ParseEmail(email string) string {
|
|
pos := strings.Index(email, "@")
|
|
parsedEmail := email[0:pos]
|
|
return parsedEmail
|
|
}
|
|
|
|
func GetMailBox(email string) (results JSONMessageHeaderInbucket, err error) {
|
|
parsedEmail := ParseEmail(email)
|
|
|
|
url := fmt.Sprintf("%s%s%s", getInbucketHost(), InbucketAPI, parsedEmail)
|
|
resp, err := http.Get(url)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
defer func() {
|
|
io.Copy(io.Discard, resp.Body)
|
|
resp.Body.Close()
|
|
}()
|
|
|
|
if resp.Body == nil {
|
|
return nil, fmt.Errorf("no mailbox")
|
|
}
|
|
|
|
var record JSONMessageHeaderInbucket
|
|
err = json.NewDecoder(resp.Body).Decode(&record)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error: %w", err)
|
|
}
|
|
if len(record) == 0 {
|
|
return nil, fmt.Errorf("no mailbox")
|
|
}
|
|
|
|
return record, nil
|
|
}
|
|
|
|
func GetMessageFromMailbox(email, id string) (JSONMessageInbucket, error) {
|
|
parsedEmail := ParseEmail(email)
|
|
|
|
var record JSONMessageInbucket
|
|
|
|
url := fmt.Sprintf("%s%s%s/%s", getInbucketHost(), InbucketAPI, parsedEmail, id)
|
|
emailResponse, err := http.Get(url)
|
|
if err != nil {
|
|
return record, err
|
|
}
|
|
defer func() {
|
|
io.Copy(io.Discard, emailResponse.Body)
|
|
emailResponse.Body.Close()
|
|
}()
|
|
|
|
if err = json.NewDecoder(emailResponse.Body).Decode(&record); err != nil {
|
|
return record, err
|
|
}
|
|
|
|
// download attachments
|
|
if len(record.Attachments) > 0 {
|
|
for i := range record.Attachments {
|
|
var bytes []byte
|
|
bytes, err = downloadAttachment(record.Attachments[i].DownloadLink)
|
|
if err != nil {
|
|
return record, err
|
|
}
|
|
record.Attachments[i].Bytes = make([]byte, len(bytes))
|
|
copy(record.Attachments[i].Bytes, bytes)
|
|
}
|
|
}
|
|
|
|
return record, err
|
|
}
|
|
|
|
func downloadAttachment(url string) ([]byte, error) {
|
|
attachmentResponse, err := http.Get(url)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer attachmentResponse.Body.Close()
|
|
|
|
buf := new(bytes.Buffer)
|
|
io.Copy(buf, attachmentResponse.Body)
|
|
return buf.Bytes(), nil
|
|
}
|
|
|
|
func DeleteMailBox(email string) (err error) {
|
|
parsedEmail := ParseEmail(email)
|
|
|
|
url := fmt.Sprintf("%s%s%s", getInbucketHost(), InbucketAPI, parsedEmail)
|
|
req, err := http.NewRequest("DELETE", url, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
client := &http.Client{}
|
|
|
|
resp, err := client.Do(req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
return nil
|
|
}
|
|
|
|
func RetryInbucket(attempts int, callback func() error) (err error) {
|
|
for i := 0; ; i++ {
|
|
err = callback()
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
if i >= (attempts - 1) {
|
|
break
|
|
}
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
fmt.Println("retrying...")
|
|
}
|
|
return fmt.Errorf("after %d attempts, last error: %s", attempts, err)
|
|
}
|
|
|
|
func getInbucketHost() (host string) {
|
|
inbucket_host := os.Getenv("CI_INBUCKET_HOST")
|
|
if inbucket_host == "" {
|
|
inbucket_host = "localhost"
|
|
}
|
|
|
|
inbucket_port := os.Getenv("CI_INBUCKET_PORT")
|
|
if inbucket_port == "" {
|
|
inbucket_port = "9001"
|
|
}
|
|
return "http://" + net.JoinHostPort(inbucket_host, inbucket_port)
|
|
}
|