소스 검색

初始提交

qdy 3 달 전
커밋
33e3e77445
3개의 변경된 파일287개의 추가작업 그리고 0개의 파일을 삭제
  1. 23
    0
      go.mod
  2. 60
    0
      go.sum
  3. 204
    0
      main.go

+ 23
- 0
go.mod 파일 보기

@@ -0,0 +1,23 @@
1
+module git.x2erp.com/qdy/x6-service-agent
2
+
3
+go 1.25.4
4
+
5
+require (
6
+	git.x2erp.com/qdy/go-base v0.1.1
7
+	git.x2erp.com/qdy/go-db v0.1.0
8
+	github.com/gorilla/mux v1.8.1
9
+)
10
+
11
+require (
12
+	filippo.io/edwards25519 v1.1.0 // indirect
13
+	github.com/go-sql-driver/mysql v1.9.3 // indirect
14
+	github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 // indirect
15
+	github.com/golang-sql/sqlexp v0.1.0 // indirect
16
+	github.com/google/uuid v1.6.0 // indirect
17
+	github.com/lib/pq v1.10.9 // indirect
18
+	github.com/microsoft/go-mssqldb v1.9.4 // indirect
19
+	github.com/sijms/go-ora/v2 v2.9.0 // indirect
20
+	golang.org/x/crypto v0.45.0 // indirect
21
+	golang.org/x/text v0.31.0 // indirect
22
+	gopkg.in/yaml.v2 v2.4.0 // indirect
23
+)

+ 60
- 0
go.sum 파일 보기

@@ -0,0 +1,60 @@
1
+filippo.io/edwards25519 v1.1.0 h1:FNf4tywRC1HmFuKW5xopWpigGjJKiJSV0Cqo0cJWDaA=
2
+filippo.io/edwards25519 v1.1.0/go.mod h1:BxyFTGdWcka3PhytdK4V28tE5sGfRvvvRV7EaN4VDT4=
3
+git.x2erp.com/qdy/go-base v0.1.1 h1:Yj6rFTCL9CShqdQeE6mUYQMugUpI25nWNL0i09SFtXo=
4
+git.x2erp.com/qdy/go-base v0.1.1/go.mod h1:+NdHouWcxqex8sJUwDTGSl/OIh5fKOT6tJteefD1MMA=
5
+git.x2erp.com/qdy/go-db v0.1.0 h1:vZhio4jZtaphpaPqV/vLmPbvU4sFrUQOwRP1yJR3K2I=
6
+git.x2erp.com/qdy/go-db v0.1.0/go.mod h1:YnHSjNaopoQEa04pu11rihPDwUedsZO/LrmyBCLx0JA=
7
+github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.0 h1:Gt0j3wceWMwPmiazCa8MzMA0MfhmPIz0Qp0FJ6qcM0U=
8
+github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.0/go.mod h1:Ot/6aikWnKWi4l9QB7qVSwa8iMphQNqkWALMoNT3rzM=
9
+github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1 h1:B+blDbyVIG3WaikNxPnhPiJ1MThR03b3vKGtER95TP4=
10
+github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1/go.mod h1:JdM5psgjfBf5fo2uWOZhflPWyDBZ/O/CNAH9CtsuZE4=
11
+github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1 h1:FPKJS1T+clwv+OLGt13a8UjqeRuh0O4SJ3lUriThc+4=
12
+github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1/go.mod h1:j2chePtV91HrC22tGoRX3sGY42uF13WzmmV80/OdVAA=
13
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1 h1:Wgf5rZba3YZqeTNJPtvqZoBu1sBN/L4sry+u2U3Y75w=
14
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1/go.mod h1:xxCBG/f/4Vbmh2XQJBsOmNdxWUY5j/s27jujKPbQf14=
15
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1 h1:bFWuoEKg+gImo7pvkiQEFAc8ocibADgXeiLAxWhWmkI=
16
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1/go.mod h1:Vih/3yc6yac2JzU4hzpaDupBJP0Flaia9rXXrU8xyww=
17
+github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2 h1:oygO0locgZJe7PpYPXT5A29ZkwJaPqcva7BVeemZOZs=
18
+github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
19
+github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
20
+github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
21
+github.com/go-sql-driver/mysql v1.9.3 h1:U/N249h2WzJ3Ukj8SowVFjdtZKfu9vlLZxjPXV1aweo=
22
+github.com/go-sql-driver/mysql v1.9.3/go.mod h1:qn46aNg1333BRMNU69Lq93t8du/dwxI64Gl8i5p1WMU=
23
+github.com/golang-jwt/jwt/v5 v5.2.2 h1:Rl4B7itRWVtYIHFrSNd7vhTiz9UpLdi6gZhZ3wEeDy8=
24
+github.com/golang-jwt/jwt/v5 v5.2.2/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
25
+github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 h1:au07oEsX2xN0ktxqI+Sida1w446QrXBRJ0nee3SNZlA=
26
+github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
27
+github.com/golang-sql/sqlexp v0.1.0 h1:ZCD6MBpcuOVfGVqsEmY5/4FtYiKz6tSyUv9LPEDei6A=
28
+github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI=
29
+github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
30
+github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
31
+github.com/gorilla/mux v1.8.1 h1:TuBL49tXwgrFYWhqrNgrUNEY92u81SPhu7sTdzQEiWY=
32
+github.com/gorilla/mux v1.8.1/go.mod h1:AKf9I4AEqPTmMytcMc0KkNouC66V3BtZ4qD5fmWSiMQ=
33
+github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
34
+github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw=
35
+github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
36
+github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
37
+github.com/microsoft/go-mssqldb v1.9.4 h1:sHrj3GcdgkxytZ09aZ3+ys72pMeyEXJowT44j74pNgs=
38
+github.com/microsoft/go-mssqldb v1.9.4/go.mod h1:GBbW9ASTiDC+mpgWDGKdm3FnFLTUsLYN3iFL90lQ+PA=
39
+github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ=
40
+github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU=
41
+github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
42
+github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
43
+github.com/sijms/go-ora/v2 v2.9.0 h1:+iQbUeTeCOFMb5BsOMgUhV8KWyrv9yjKpcK4x7+MFrg=
44
+github.com/sijms/go-ora/v2 v2.9.0/go.mod h1:QgFInVi3ZWyqAiJwzBQA+nbKYKH77tdp1PYoCqhR2dU=
45
+github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
46
+github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
47
+golang.org/x/crypto v0.45.0 h1:jMBrvKuj23MTlT0bQEOBcAE0mjg8mK9RXFhRH6nyF3Q=
48
+golang.org/x/crypto v0.45.0/go.mod h1:XTGrrkGJve7CYK7J8PEww4aY7gM3qMCElcJQ8n8JdX4=
49
+golang.org/x/net v0.47.0 h1:Mx+4dIFzqraBXUugkia1OOvlD6LemFo1ALMHjrXDOhY=
50
+golang.org/x/net v0.47.0/go.mod h1:/jNxtkgq5yWUGYkaZGqo27cfGZ1c5Nen03aYrrKpVRU=
51
+golang.org/x/sys v0.38.0 h1:3yZWxaJjBmCWXqhN1qh02AkOnCQ1poK6oF+a7xWL6Gc=
52
+golang.org/x/sys v0.38.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
53
+golang.org/x/text v0.31.0 h1:aC8ghyu4JhP8VojJ2lEHBnochRno1sgL6nEi9WGFGMM=
54
+golang.org/x/text v0.31.0/go.mod h1:tKRAlv61yKIjGGHX/4tP1LTbc13YSec1pxVEWXzfoeM=
55
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
56
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
57
+gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
58
+gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
59
+gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
60
+gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=

+ 204
- 0
main.go 파일 보기

@@ -0,0 +1,204 @@
1
+package main
2
+
3
+import (
4
+	"database/sql"
5
+	"encoding/json"
6
+	"fmt"
7
+	"log"
8
+	"net/http"
9
+	"time"
10
+
11
+	"git.x2erp.com/qdy/go-base/types"
12
+	// 注意:这里要使用 factory 包的正确导入路径(和你原代码一致)
13
+	"git.x2erp.com/qdy/go-db/factory"
14
+	"github.com/gorilla/mux"
15
+)
16
+
17
+// 全局变量,只初始化一次(复用原逻辑,确保工厂和DB连接单例)
18
+var dbFactory *factory.DBFactory
19
+var db *sql.DB
20
+
21
+// QueryRequest 请求结构体(接收前端SQL参数)
22
+type QueryRequest struct {
23
+	SQL string `json:"sql"`
24
+}
25
+
26
+func main() {
27
+	var err error
28
+
29
+	// 1. 创建数据库工厂(只执行一次)
30
+	dbFactory, err = factory.NewDBFactory()
31
+	if err != nil {
32
+		log.Fatalf("Failed to create DB factory: %v", err)
33
+	}
34
+
35
+	// 2. 显示基础信息(保留原日志输出)
36
+	drivers := dbFactory.GetAvailableDrivers()
37
+	fmt.Printf("Available database drivers: %v\n", drivers)
38
+
39
+	config := dbFactory.GetConfig()
40
+	fmt.Printf("Using database type: %s\n", config.Database.Type)
41
+	fmt.Printf("Database host: %s:%d\n", config.Database.Host, config.Database.Port)
42
+	fmt.Printf("Database name: %s\n", config.Database.Database)
43
+
44
+	// 3. 创建数据库连接(全局复用)
45
+	db, err = dbFactory.CreateDB()
46
+	if err != nil {
47
+		log.Fatalf("Failed to create database connection: %v", err)
48
+	}
49
+	defer db.Close()
50
+
51
+	// 4. 测试连接(保留原校验逻辑)
52
+	if err := testConnection(db, config.Database.Type); err != nil {
53
+		log.Fatalf("Database connection test failed: %v", err)
54
+	} else {
55
+		fmt.Println("Database connection test passed!")
56
+	}
57
+
58
+	// 5. 启动HTTP服务
59
+	startHTTPServer()
60
+}
61
+
62
+// 启动HTTP服务器(简化,直接使用全局DB连接)
63
+func startHTTPServer() {
64
+	router := mux.NewRouter()
65
+
66
+	// 核心路由:SQL查询(POST)- 直接返回 types.QueryResult
67
+	router.HandleFunc("/api/query", queryHandler).Methods("POST")
68
+	// 辅助路由:健康检查、数据库信息(保留原功能)
69
+	router.HandleFunc("/api/health", healthHandler).Methods("GET")
70
+	router.HandleFunc("/api/info", infoHandler).Methods("GET")
71
+
72
+	// 服务器配置(保留原超时设置)
73
+	server := &http.Server{
74
+		Addr:         ":8080",
75
+		Handler:      router,
76
+		ReadTimeout:  30 * time.Second,
77
+		WriteTimeout: 30 * time.Second,
78
+		IdleTimeout:  60 * time.Second,
79
+	}
80
+
81
+	// 启动日志(保留原格式)
82
+	fmt.Println("Database microservice starting on :8080")
83
+	fmt.Println("Endpoints:")
84
+	fmt.Println("  POST /api/query  - Execute SQL query (return types.QueryResult)")
85
+	fmt.Println("  GET  /api/health - Health check")
86
+	fmt.Println("  GET  /api/info   - Database info")
87
+
88
+	log.Fatal(server.ListenAndServe())
89
+}
90
+
91
+// queryHandler SQL查询处理(核心修改)
92
+// 1. 修复 factory.QuickQueryToJSON 调用(确保导入路径正确)
93
+// 2. 直接返回 types.QueryResult,不二次封装
94
+func queryHandler(w http.ResponseWriter, r *http.Request) {
95
+	w.Header().Set("Content-Type", "application/json")
96
+
97
+	// 1. 解析请求参数
98
+	var req QueryRequest
99
+	if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
100
+		// 参数错误时,返回 types.QueryResult 格式的错误响应
101
+		json.NewEncoder(w).Encode(&types.QueryResult{
102
+			Success: false,
103
+			Error:   fmt.Sprintf("Invalid request body: %v", err),
104
+			Data:    nil,
105
+		})
106
+		return
107
+	}
108
+
109
+	// 2. 校验SQL非空
110
+	if req.SQL == "" {
111
+		json.NewEncoder(w).Encode(&types.QueryResult{
112
+			Success: false,
113
+			Error:   "SQL statement cannot be empty",
114
+			Data:    nil,
115
+		})
116
+		return
117
+	}
118
+
119
+	// 3. 核心逻辑:调用 factory.QuickQueryToJSON(确保工厂包导出了该方法)
120
+	// 注意:如果仍提示 undefined,检查 factory 包是否真的导出了 QuickQueryToJSON(首字母大写)
121
+	result := factory.QuickQueryToJSON(db, req.SQL)
122
+
123
+	// 4. 直接返回结果(types.QueryResult 原生格式)
124
+	json.NewEncoder(w).Encode(result)
125
+}
126
+
127
+// healthHandler 健康检查(修复语法错误,保持返回格式统一)
128
+func healthHandler(w http.ResponseWriter, r *http.Request) {
129
+	w.Header().Set("Content-Type", "application/json")
130
+
131
+	// 校验DB连接状态
132
+	err := db.Ping()
133
+	success := err == nil
134
+
135
+	// 标准 if-else 赋值状态
136
+	var status string
137
+	if success {
138
+		status = "UP"
139
+	} else {
140
+		status = "DOWN"
141
+	}
142
+
143
+	// 返回 types.QueryResult 格式(和查询接口保持一致)
144
+	json.NewEncoder(w).Encode(&types.QueryResult{
145
+		Success: success,
146
+		Data: map[string]interface{}{
147
+			"status":   status,
148
+			"time":     time.Now().Format(time.RFC3339),
149
+			"database": dbFactory.GetConfig().Database.Type,
150
+		},
151
+		Error: func() string {
152
+			if err != nil {
153
+				return err.Error()
154
+			}
155
+			return ""
156
+		}(),
157
+	})
158
+}
159
+
160
+// infoHandler 数据库信息(返回 types.QueryResult 格式)
161
+func infoHandler(w http.ResponseWriter, r *http.Request) {
162
+	w.Header().Set("Content-Type", "application/json")
163
+
164
+	config := dbFactory.GetConfig()
165
+	drivers := dbFactory.GetAvailableDrivers()
166
+
167
+	// 直接返回 types.QueryResult 格式
168
+	json.NewEncoder(w).Encode(&types.QueryResult{
169
+		Success: true,
170
+		Data: map[string]interface{}{
171
+			"database_type":     config.Database.Type,
172
+			"database_host":     fmt.Sprintf("%s:%d", config.Database.Host, config.Database.Port),
173
+			"database_name":     config.Database.Database,
174
+			"available_drivers": drivers,
175
+			"service_time":      time.Now().Format(time.RFC3339),
176
+		},
177
+		Error: "",
178
+	})
179
+}
180
+
181
+// testConnection 测试数据库连接(保留原逻辑)
182
+func testConnection(db *sql.DB, dbType string) error {
183
+	var query string
184
+	switch dbType {
185
+	case "mysql", "postgres", "sqlserver":
186
+		query = "SELECT 1"
187
+	case "oracle":
188
+		query = "SELECT 1 FROM DUAL"
189
+	default:
190
+		query = "SELECT 1"
191
+	}
192
+
193
+	var result int
194
+	err := db.QueryRow(query).Scan(&result)
195
+	if err != nil {
196
+		return fmt.Errorf("test query failed: %v", err)
197
+	}
198
+
199
+	if result != 1 {
200
+		return fmt.Errorf("unexpected test result: %d", result)
201
+	}
202
+
203
+	return nil
204
+}

Loading…
취소
저장