Full Mattermost server source with integrated Community Enterprise features. Includes vendor directory for offline/air-gapped builds. Structure: - enterprise-impl/: Enterprise feature implementations - enterprise-community/: Init files that register implementations - enterprise/: Bridge imports (community_imports.go) - vendor/: All dependencies for offline builds Build (online): go build ./cmd/mattermost Build (offline/air-gapped): go build -mod=vendor ./cmd/mattermost 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
193 lines
4.3 KiB
Go
193 lines
4.3 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package app
|
|
|
|
import (
|
|
"archive/tar"
|
|
"bytes"
|
|
"compress/gzip"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func assertDirectoryContents(t *testing.T, dir string, expectedFiles []string) {
|
|
var files []string
|
|
err := filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
|
require.NoError(t, err)
|
|
file := strings.TrimPrefix(path, dir)
|
|
file = strings.TrimPrefix(file, "/")
|
|
files = append(files, file)
|
|
return nil
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
sort.Strings(files)
|
|
sort.Strings(expectedFiles)
|
|
assert.Equal(t, expectedFiles, files)
|
|
}
|
|
|
|
func TestExtractTarGz(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
makeArchive := func(t *testing.T, files []*tar.Header) bytes.Buffer {
|
|
// Build an in-memory archive with the specified files, writing the path as each
|
|
// file's contents when applicable.
|
|
var archive bytes.Buffer
|
|
archiveGzWriter := gzip.NewWriter(&archive)
|
|
archiveWriter := tar.NewWriter(archiveGzWriter)
|
|
for _, file := range files {
|
|
if file.Typeflag == tar.TypeReg {
|
|
contents := []byte(file.Name)
|
|
file.Size = int64(len(contents))
|
|
err := archiveWriter.WriteHeader(file)
|
|
require.NoError(t, err)
|
|
|
|
var written int
|
|
written, err = archiveWriter.Write(contents)
|
|
require.NoError(t, err)
|
|
require.EqualValues(t, len(contents), written)
|
|
} else {
|
|
err := archiveWriter.WriteHeader(file)
|
|
require.NoError(t, err)
|
|
}
|
|
}
|
|
err := archiveWriter.Close()
|
|
require.NoError(t, err)
|
|
err = archiveGzWriter.Close()
|
|
require.NoError(t, err)
|
|
|
|
return archive
|
|
}
|
|
|
|
t.Run("empty dst", func(t *testing.T) {
|
|
archive := makeArchive(t, nil)
|
|
err := extractTarGz(&archive, "")
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("huge tar", func(t *testing.T) {
|
|
files := make([]*tar.Header, 0, 10000)
|
|
for i := range 10000 {
|
|
files = append(files, &tar.Header{
|
|
Name: fmt.Sprintf("%d.txt", i),
|
|
Typeflag: tar.TypeReg,
|
|
})
|
|
}
|
|
|
|
dst, err := os.MkdirTemp("", "TestExtractTarGz")
|
|
require.NoError(t, err)
|
|
defer os.RemoveAll(dst)
|
|
|
|
archive := makeArchive(t, files)
|
|
err = extractTarGz(&archive, dst)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
testCases := []struct {
|
|
Files []*tar.Header
|
|
ExpectedError bool
|
|
ExpectedFiles []string
|
|
}{
|
|
{
|
|
[]*tar.Header{{Name: "../test/path", Typeflag: tar.TypeDir}},
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
[]*tar.Header{{Name: "../../test/path", Typeflag: tar.TypeDir}},
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
[]*tar.Header{{Name: "../../test/../path", Typeflag: tar.TypeDir}},
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
[]*tar.Header{{Name: "test/../../path", Typeflag: tar.TypeDir}},
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
[]*tar.Header{{Name: "test/path/../..", Typeflag: tar.TypeDir}},
|
|
false,
|
|
[]string{""},
|
|
},
|
|
{
|
|
[]*tar.Header{{Name: "test", Typeflag: tar.TypeDir}},
|
|
false,
|
|
[]string{"", "test"},
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "test", Typeflag: tar.TypeDir},
|
|
{Name: "test/path", Typeflag: tar.TypeDir},
|
|
},
|
|
false,
|
|
[]string{"", "test", "test/path"},
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "test", Typeflag: tar.TypeDir},
|
|
{Name: "test/path/", Typeflag: tar.TypeDir},
|
|
},
|
|
false,
|
|
[]string{"", "test", "test/path"},
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "test", Typeflag: tar.TypeDir},
|
|
{Name: "test/path", Typeflag: tar.TypeDir},
|
|
{Name: "test/path/file.ext", Typeflag: tar.TypeReg},
|
|
},
|
|
false,
|
|
[]string{"", "test", "test/path", "test/path/file.ext"},
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "/../../file.ext", Typeflag: tar.TypeReg},
|
|
},
|
|
true,
|
|
nil,
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "/../../link", Typeflag: tar.TypeLink},
|
|
},
|
|
false,
|
|
[]string{""},
|
|
},
|
|
{
|
|
[]*tar.Header{
|
|
{Name: "..file", Typeflag: tar.TypeReg},
|
|
},
|
|
false,
|
|
[]string{"", "..file"},
|
|
},
|
|
}
|
|
|
|
for i, testCase := range testCases {
|
|
t.Run(fmt.Sprintf("test-%d", i), func(t *testing.T) {
|
|
dst, err := os.MkdirTemp("", "TestExtractTarGz")
|
|
require.NoError(t, err)
|
|
defer os.RemoveAll(dst)
|
|
|
|
archive := makeArchive(t, testCase.Files)
|
|
err = extractTarGz(&archive, dst)
|
|
if testCase.ExpectedError {
|
|
require.Error(t, err)
|
|
} else {
|
|
require.NoError(t, err)
|
|
assertDirectoryContents(t, dst, testCase.ExpectedFiles)
|
|
}
|
|
})
|
|
}
|
|
}
|