mirror of
https://github.com/openimsdk/open-im-server.git
synced 2026-04-28 14:29:19 +08:00
☀️ feat: Enhancing OpenIM with Integrated E2E Testing and CI/CD Enhancements (#1359)
* cicd: robot automated Change * feat: add api test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add api test make file Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add openim e2e test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add openim e2e test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * fix: Fixed some unused scripts and some names Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * docs: optimize openim docs Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add prom address Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add openim info test * feat: add openim images config path Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * fix: fix tim file rename * fix: fix tim file rename * fix: fix tim file rename * fix: fix tim file rename * fix: add openim test e2e * feat: add openim test .keep Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: add openim test .keep Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: openim test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: openim test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> * feat: openim test Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> --------- Signed-off-by: Xinwei Xiong(cubxxw) <3293172751nss@gmail.com> Co-authored-by: cubxxw <cubxxw@users.noreply.github.com>
This commit is contained in:
@@ -0,0 +1 @@
|
||||
.keep
|
||||
@@ -0,0 +1,21 @@
|
||||
package config
|
||||
|
||||
import "flag"
|
||||
|
||||
// Flags is the flag set that AddOptions adds to. Test authors should
|
||||
// also use it instead of directly adding to the global command line.
|
||||
var Flags = flag.NewFlagSet("", flag.ContinueOnError)
|
||||
|
||||
// CopyFlags ensures that all flags that are defined in the source flag
|
||||
// set appear in the target flag set as if they had been defined there
|
||||
// directly. From the flag package it inherits the behavior that there
|
||||
// is a panic if the target already contains a flag from the source.
|
||||
func CopyFlags(source *flag.FlagSet, target *flag.FlagSet) {
|
||||
source.VisitAll(func(flag *flag.Flag) {
|
||||
// We don't need to copy flag.DefValue. The original
|
||||
// default (from, say, flag.String) was stored in
|
||||
// the value and gets extracted by Var for the help
|
||||
// message.
|
||||
target.Var(flag.Value, flag.Name, flag.Usage)
|
||||
})
|
||||
}
|
||||
@@ -0,0 +1,75 @@
|
||||
package config
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"reflect"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestCopyFlags(t *testing.T) {
|
||||
type args struct {
|
||||
source *flag.FlagSet
|
||||
target *flag.FlagSet
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
wantErr bool
|
||||
}{
|
||||
{
|
||||
name: "Copy empty source to empty target",
|
||||
args: args{
|
||||
source: flag.NewFlagSet("source", flag.ContinueOnError),
|
||||
target: flag.NewFlagSet("target", flag.ContinueOnError),
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "Copy non-empty source to empty target",
|
||||
args: args{
|
||||
source: func() *flag.FlagSet {
|
||||
fs := flag.NewFlagSet("source", flag.ContinueOnError)
|
||||
fs.String("test-flag", "default", "test usage")
|
||||
return fs
|
||||
}(),
|
||||
target: flag.NewFlagSet("target", flag.ContinueOnError),
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "Copy source to target with existing flag",
|
||||
args: args{
|
||||
source: func() *flag.FlagSet {
|
||||
fs := flag.NewFlagSet("source", flag.ContinueOnError)
|
||||
fs.String("test-flag", "default", "test usage")
|
||||
return fs
|
||||
}(),
|
||||
target: func() *flag.FlagSet {
|
||||
fs := flag.NewFlagSet("target", flag.ContinueOnError)
|
||||
fs.String("test-flag", "default", "test usage")
|
||||
return fs
|
||||
}(),
|
||||
},
|
||||
wantErr: true,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
defer func() {
|
||||
if r := recover(); (r != nil) != tt.wantErr {
|
||||
t.Errorf("CopyFlags() panic = %v, wantErr %v", r, tt.wantErr)
|
||||
}
|
||||
}()
|
||||
CopyFlags(tt.args.source, tt.args.target)
|
||||
|
||||
// 验证复制的标记
|
||||
if !tt.wantErr {
|
||||
tt.args.source.VisitAll(func(f *flag.Flag) {
|
||||
if gotFlag := tt.args.target.Lookup(f.Name); gotFlag == nil || !reflect.DeepEqual(gotFlag, f) {
|
||||
t.Errorf("CopyFlags() failed to copy flag %s", f.Name)
|
||||
}
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
.keep
|
||||
@@ -0,0 +1 @@
|
||||
package ginkgowrapper
|
||||
@@ -0,0 +1 @@
|
||||
package ginkgowrapper
|
||||
@@ -0,0 +1 @@
|
||||
.keep
|
||||
@@ -0,0 +1,152 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"os"
|
||||
"os/exec"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
var (
|
||||
// The default template version
|
||||
defaultTemplateVersion = "v1.3.0"
|
||||
)
|
||||
|
||||
func main() {
|
||||
// Define the URL to get the latest version
|
||||
// latestVersionURL := "https://github.com/openimsdk/chat/releases/latest"
|
||||
// latestVersion, err := getLatestVersion(latestVersionURL)
|
||||
// if err != nil {
|
||||
// fmt.Printf("Failed to get the latest version: %v\n", err)
|
||||
// return
|
||||
// }
|
||||
latestVersion := defaultTemplateVersion
|
||||
|
||||
// Construct the download URL
|
||||
downloadURL := fmt.Sprintf("https://github.com/openimsdk/chat/releases/download/%s/chat_Linux_x86_64.tar.gz", latestVersion)
|
||||
|
||||
// Set the installation directory
|
||||
installDir := "/tmp/chat"
|
||||
|
||||
// Clear the installation directory before proceeding
|
||||
err := os.RemoveAll(installDir)
|
||||
if err != nil {
|
||||
fmt.Printf("Failed to clear installation directory: %v\n", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Create the installation directory
|
||||
err = os.MkdirAll(installDir, 0755)
|
||||
if err != nil {
|
||||
fmt.Printf("Failed to create installation directory: %v\n", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Download and extract OpenIM Chat to the installation directory
|
||||
err = downloadAndExtract(downloadURL, installDir)
|
||||
if err != nil {
|
||||
fmt.Printf("Failed to download and extract OpenIM Chat: %v\n", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Create configuration file directory
|
||||
configDir := filepath.Join(installDir, "config")
|
||||
err = os.MkdirAll(configDir, 0755)
|
||||
if err != nil {
|
||||
fmt.Printf("Failed to create configuration directory: %v\n", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Download configuration files
|
||||
configURL := "https://raw.githubusercontent.com/openimsdk/chat/main/config/config.yaml"
|
||||
err = downloadAndExtract(configURL, configDir)
|
||||
if err != nil {
|
||||
fmt.Printf("Failed to download and extract configuration files: %v\n", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Define the processes to be started
|
||||
cmds := []string{
|
||||
"admin-api",
|
||||
"admin-rpc",
|
||||
"chat-api",
|
||||
"chat-rpc",
|
||||
}
|
||||
|
||||
// Start each process in a new goroutine
|
||||
for _, cmd := range cmds {
|
||||
go startProcess(filepath.Join(installDir, cmd))
|
||||
}
|
||||
|
||||
// Block the main thread indefinitely
|
||||
select {}
|
||||
}
|
||||
|
||||
// getLatestVersion fetches the latest version number from a given URL
|
||||
func getLatestVersion(url string) (string, error) {
|
||||
resp, err := http.Get(url)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
location := resp.Header.Get("Location")
|
||||
if location == "" {
|
||||
return defaultTemplateVersion, nil
|
||||
}
|
||||
|
||||
// Extract the version number from the URL
|
||||
latestVersion := filepath.Base(location)
|
||||
return latestVersion, nil
|
||||
}
|
||||
|
||||
// downloadAndExtract downloads a file from a URL and extracts it to a destination directory
|
||||
func downloadAndExtract(url, destDir string) error {
|
||||
resp, err := http.Get(url)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
return fmt.Errorf("error downloading file, HTTP status code: %d", resp.StatusCode)
|
||||
}
|
||||
|
||||
// Create the destination directory
|
||||
err = os.MkdirAll(destDir, 0755)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Define the path for the downloaded file
|
||||
filePath := filepath.Join(destDir, "downloaded_file.tar.gz")
|
||||
file, err := os.Create(filePath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer file.Close()
|
||||
|
||||
// Copy the downloaded file
|
||||
_, err = io.Copy(file, resp.Body)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// Extract the file
|
||||
cmd := exec.Command("tar", "xzvf", filePath, "-C", destDir)
|
||||
cmd.Stdout = os.Stdout
|
||||
cmd.Stderr = os.Stderr
|
||||
return cmd.Run()
|
||||
}
|
||||
|
||||
// startProcess starts a process and prints any errors encountered
|
||||
func startProcess(cmdPath string) {
|
||||
cmd := exec.Command(cmdPath)
|
||||
cmd.Stdout = os.Stdout
|
||||
cmd.Stderr = os.Stderr
|
||||
if err := cmd.Run(); err != nil {
|
||||
fmt.Printf("Failed to start process %s: %v\n", cmdPath, err)
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user