☀️ 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:
Xinwei Xiong
2023-11-10 19:37:25 +08:00
committed by GitHub
parent 686fa80800
commit e2004c1e9d
154 changed files with 4020 additions and 928 deletions
+1
View File
@@ -0,0 +1 @@
.keep
+21
View File
@@ -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)
})
}
+75
View File
@@ -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)
}
})
}
})
}
}
+1
View File
@@ -0,0 +1 @@
.keep
@@ -0,0 +1 @@
package ginkgowrapper
@@ -0,0 +1 @@
package ginkgowrapper
+1
View File
@@ -0,0 +1 @@
.keep
+152
View File
@@ -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)
}
}