GodoOS当地代理实现详解

打印 上一主题 下一主题

主题 893|帖子 893|积分 2679

弁言

  • 在现代软件开发中,代理服务器饰演着至关告急的角色。它们可以用于负载均衡、请求转发、缓存、安全控制等多种场景。本文将详细介绍 godoos 中的当地代理实现,包括其架构设计、核心功能以及具体的实现细节。
架构设计

核心组件
ProxyServer 结构体:用于存储服务类型和现实服务对象。
FileServer 结构体:用于存储文件静态服务的信息。
sync.Map:用于全局存储代理服务映射,确保线程安全。
主要功能
HTTP 代理:将请求转发到指定的远程服务器。
UDP 代理:实现 UDP 数据包的转发。
文件静态服务:提供静态文件的访问。
核心功能实现
1. 创建当地代理

CreateLocalProxyHandler 函数用于创建一个新的当地代理。它从 HTTP 请求中剖析代理设置,并将其存储到数据库中。然后启动相应的代理服务。
  1. func CreateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
  2.         var lp model.LocalProxy
  3.         err := json.NewDecoder(r.Body).Decode(&lp)
  4.         if err != nil {
  5.                 http.Error(w, err.Error(), http.StatusBadRequest)
  6.                 return
  7.         }
  8.         err = model.Db.Create(&lp).Error
  9.         if err != nil {
  10.                 http.Error(w, err.Error(), http.StatusInternalServerError)
  11.                 return
  12.         }
  13.         // 启动代理服务
  14.         go startProxy(lp)
  15.         libs.SuccessMsg(w, lp, "")
  16. }
复制代码
2. 获取当地代理

GetLocalProxiesHandler 和 GetLocalProxyHandler 函数分别用于获取所有当地代理和单个当地代理的设置信息。
  1. func GetLocalProxiesHandler(w http.ResponseWriter, r *http.Request) {
  2. // 获取查询参数 page 和 limit
  3. pageStr := r.URL.Query().Get("page")
  4. limitStr := r.URL.Query().Get("limit")
  5. page, err := strconv.Atoi(pageStr)
  6. if err != nil || page < 1 {
  7.         page = 1
  8. }
  9. limit, err := strconv.Atoi(limitStr)
  10. if err != nil || limit < 1 {
  11.         limit = 10
  12. }
  13. // 定义响应结构体
  14. type ProxyResponse struct {
  15.         Proxies []model.LocalProxy `json:"proxies"`
  16.         Total   int64              `json:"total"`
  17. }
  18. // 修改处理函数
  19. proxies, total, err := model.GetLocalProxies(page, limit)
  20. if err != nil {
  21.         http.Error(w, err.Error(), http.StatusInternalServerError)
  22.         return
  23. }
  24. libs.SuccessMsg(w, ProxyResponse{Proxies: proxies, Total: total}, "")
  25. }
复制代码
UpdateLocalProxyHandler 函数用于更新当地代理的设置。更新后,会制止旧的代理服务并启动新的代理服务。
  1. gofunc UpdateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
  2. var lp model.LocalProxy
  3. err := json.NewDecoder(r.Body).Decode(&lp)
  4. if err != nil {
  5. http.Error(w, err.Error(), http.StatusBadRequest)
  6. return
  7. }
  8. err = model.Db.Model(&model.LocalProxy{}).Where("id = ?", lp.ID).Updates(lp).Error
  9. if err != nil {
  10.         http.Error(w, err.Error(), http.StatusInternalServerError)
  11.         return
  12. }
  13. // 停止旧的代理服务
  14. stopProxy(lp.ID)
  15. // 启动新的代理服务
  16. go startProxy(lp)
  17. libs.SuccessMsg(w, lp, "")
  18. }
复制代码
DeleteLocalProxyHandler 函数用于删除当地代理。删除前会制止相应的代理服务。
  1. gofunc DeleteLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
  2. idStr := r.URL.Query().Get("id")
  3. id, err := strconv.Atoi(idStr)
  4. if err != nil {
  5. http.Error(w, "Invalid ID", http.StatusBadRequest)
  6. return
  7. }
  8. // 停止代理服务
  9. stopProxy(uint(id))
  10. err = model.Db.Delete(&model.LocalProxy{}, uint(id)).Error
  11. if err != nil {
  12.         http.Error(w, err.Error(), http.StatusInternalServerError)
  13.         return
  14. }
  15. libs.SuccessMsg(w, nil, "delete proxy success")
  16. }
复制代码
HandlerSetProxyStatus 函数用于启用或禁用代理服务。根据代理的状态,启动或制止相应的代理服务。
  1. gofunc HandlerSetProxyStatus(w http.ResponseWriter, r *http.Request) {
  2. idStr := r.URL.Query().Get("id")
  3. if idStr == "" {
  4. libs.ErrorMsg(w, "id is empty")
  5. return
  6. }
  7. id, err := strconv.Atoi(idStr)
  8. if err != nil || id == 0 {
  9. libs.ErrorMsg(w, "id is not number")
  10. return
  11. }
  12. var proxy model.LocalProxy
  13. if err := model.Db.First(&proxy, uint(id)).Error; err != nil {
  14. libs.ErrorMsg(w, "proxy not found")
  15. return
  16. }
  17. if err := model.Db.Model(&model.LocalProxy{}).Where("id = ?", proxy.ID).Update("status", !proxy.Status).Error; err != nil {
  18. libs.ErrorMsg(w, "update proxy status failed")
  19. return
  20. }
  21. if !proxy.Status {
  22. startProxy(proxy)
  23. } else {
  24. stopProxy(proxy.ID)
  25. }
  26. libs.SuccessMsg(w, nil, "")
  27. }
复制代码
启动代理服务

startProxy 函数根据代理的类型启动相应的代理服务。
  1. func startProxy(proxy model.LocalProxy) {
  2. switch proxy.ProxyType {
  3. case "http":
  4. go func(p model.LocalProxy) {
  5. httpProxyHandler(p)
  6. }(proxy)
  7. case "udp":
  8. go func(p model.LocalProxy) {
  9. udpProxyHandler(p)
  10. }(proxy)
  11. case "file":
  12. go func(p model.LocalProxy) {
  13. fileServerHandler(p)
  14. }(proxy)
  15. default:
  16. fmt.Printf("Unknown proxy type: %s\n", proxy.ProxyType)
  17. }
  18. }
复制代码
stopProxy 函数根据代理的类型制止相应的代理服务。
  1. func stopProxy(id uint) {
  2. if server, ok := proxyServers.Load(id); ok {
  3. proxyServer, ok := server.(ProxyServer)
  4. if !ok {
  5. fmt.Printf("Failed to load proxy server for ID %d\n", id)
  6. return
  7. }
  8.         switch proxyServer.Type {
  9.         case "http":
  10.                 httpServer, ok := proxyServer.Server.(*http.Server)
  11.                 if ok {
  12.                         // 创建一个上下文,用于传递给 server.Shutdown(ctx)
  13.                         ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  14.                         defer cancel()
  15.                         if err := httpServer.Shutdown(ctx); err != nil {
  16.                                 fmt.Printf("Failed to shutdown HTTP server on port %s: %v\n", httpServer.Addr, err)
  17.                         } else {
  18.                                 fmt.Printf("Stopped HTTP server on port %s\n", httpServer.Addr)
  19.                         }
  20.                 }
  21.         case "udp":
  22.                 udpConn, ok := proxyServer.Server.(*net.UDPConn)
  23.                 if ok {
  24.                         udpConn.Close()
  25.                         fmt.Printf("Stopped UDP server on port %d\n", udpConn.LocalAddr().(*net.UDPAddr).Port)
  26.                 }
  27.         case "file":
  28.                 fileServer, ok := proxyServer.Server.(*FileServer)
  29.                 if ok {
  30.                         // 创建一个上下文,用于传递给 server.Shutdown(ctx)
  31.                         ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  32.                         defer cancel()
  33.                         if err := fileServer.Server.Shutdown(ctx); err != nil {
  34.                                 fmt.Printf("Failed to shutdown file server on port %d: %v\n", fileServer.Port, err)
  35.                         } else {
  36.                                 fmt.Printf("Stopped file server on port %d\n", fileServer.Port)
  37.                         }
  38.                 }
  39.         default:
  40.                 fmt.Printf("Unknown proxy type: %s\n", proxyServer.Type)
  41.         }
  42.         proxyServers.Delete(id)
  43. }
  44. }
复制代码
HTTP 代理处理函数

httpProxyHandler 函数用于处理 HTTP 代理请求,将请求转发到指定的远程服务器。
  1. func httpProxyHandler(proxy model.LocalProxy) {
  2. remote, err := url.Parse(proxy.Domain)
  3. if err != nil {
  4. fmt.Printf("Failed to parse remote URL for port %d: %v\n", proxy.Port, err)
  5. return
  6. }
  7. if remote.Scheme == "" {
  8.         fmt.Printf("Remote URL for port %d does not contain a scheme (http/https): %s\n", proxy.Port, proxy.Domain)
  9.         return
  10. }
  11. reverseProxy := httputil.NewSingleHostReverseProxy(remote)
  12. // 启动 HTTP 服务器并监听指定端口
  13. server := &http.Server{
  14.         Addr:    fmt.Sprintf(":%d", proxy.Port),
  15.         Handler: reverseProxy,
  16. }
  17. proxyServers.Store(proxy.ID, ProxyServer{Type: "http", Server: server})
  18. fmt.Printf("Starting HTTP proxy on port %d and forwarding to %s:%d\n", proxy.Port, proxy.Domain, proxy.Port)
  19. if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
  20.         fmt.Printf("Failed to start HTTP proxy on port %d: %v\n", proxy.Port, err)
  21. }
  22. }
复制代码
udpProxyHandler 函数用于处理 UDP 代理请求,将数据包转发到指定的远程服务器。
  1. func udpProxyHandler(proxy model.LocalProxy) {
  2. localAddr, err := net.ResolveUDPAddr("udp", fmt.Sprintf(":%d", proxy.Port))
  3. if err != nil {
  4. fmt.Printf("Failed to resolve local UDP address for port %d: %v\n", proxy.Port, err)
  5. return
  6. }
  7. remoteAddr, err := net.ResolveUDPAddr("udp", proxy.Domain)
  8. if err != nil {
  9.         fmt.Printf("Failed to resolve remote UDP address for port %d: %v\n", proxy.Port, err)
  10.         return
  11. }
  12. conn, err := net.ListenUDP("udp", localAddr)
  13. if err != nil {
  14.         fmt.Printf("Failed to listen on UDP port %d: %v\n", proxy.Port, err)
  15.         return
  16. }
  17. defer conn.Close()
  18. proxyServers.Store(proxy.ID, ProxyServer{Type: "udp", Server: conn})
  19. buffer := make([]byte, 1024)
  20. for {
  21.         n, addr, err := conn.ReadFromUDP(buffer)
  22.         if err != nil {
  23.                 fmt.Printf("Failed to read from UDP: %v\n", err)
  24.                 continue
  25.         }
  26.         remoteConn, err := net.DialUDP("udp", nil, remoteAddr)
  27.         if err != nil {
  28.                 fmt.Printf("Failed to dial remote UDP: %v\n", err)
  29.                 continue
  30.         }
  31.         _, err = remoteConn.Write(buffer[:n])
  32.         if err != nil {
  33.                 fmt.Printf("Failed to write to remote UDP: %v\n", err)
  34.                 continue
  35.         }
  36.         // 读取远程服务器的响应并转发回客户端
  37.         n, err = remoteConn.Read(buffer)
  38.         if err != nil {
  39.                 fmt.Printf("Failed to read from remote UDP: %v\n", err)
  40.                 continue
  41.         }
  42.         _, err = conn.WriteToUDP(buffer[:n], addr)
  43.         if err != nil {
  44.                 fmt.Printf("Failed to write to client UDP: %v\n", err)
  45.                 continue
  46.         }
  47. }
  48. }
复制代码
fileServerHandler 函数用于提供静态文件的访问。
  1. func fileServerHandler(proxy model.LocalProxy) {
  2. // 创建文件服务器的 HTTP 处理函数
  3. fileHandler := http.FileServer(http.Dir(proxy.Path))
  4. // 启动 HTTP 服务器并监听指定端口
  5. server := &http.Server{
  6.         Addr:    fmt.Sprintf(":%d", proxy.Port),
  7.         Handler: fileHandler,
  8. }
  9. proxyServers.Store(proxy.ID, ProxyServer{Type: "file", Server: &FileServer{Port: int(proxy.Port), Server: server}})
  10. fmt.Printf("Starting file server on port %d serving files from %s\n", proxy.Port, proxy.Path)
  11. if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
  12.         fmt.Printf("Failed to start file server on port %d: %v\n", proxy.Port, err)
  13. }
  14. }
复制代码
通过本文的介绍,我们详细地了解了 godoos 中当地代理的实现。godoos 通过灵活的代理类型支持 HTTP、UDP 和文件静态服务,并提供了丰富的管理功能,包括创建、更新、删除和设置代理状态。这种设计使得 godoos 可以大概适应各种复杂的网络环境和需求。
盼望本文能资助你更好地理解和利用 godoos 中的当地代理功能。如果你有任何题目或发起,欢迎在评论区留言交流。
全部代码
  1. package proxyimport (        "context"        "encoding/json"        "fmt"        "godo/libs"        "godo/model"        "net"        "net/http"        "net/http/httputil"        "net/url"        "strconv"        "sync"        "time")// 全局代理服务映射var proxyServers sync.Map// ProxyServer 结构体用于存储服务类型和现实服务对象type ProxyServer struct {        Type   string        Server interface{}}// FileServer 结构体用于存储文件静态服务的信息type FileServer struct {        Port   int        Server *http.Server}// 创建 LocalProxy 的 HTTP 处理函数func CreateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {
  2.         var lp model.LocalProxy
  3.         err := json.NewDecoder(r.Body).Decode(&lp)
  4.         if err != nil {
  5.                 http.Error(w, err.Error(), http.StatusBadRequest)
  6.                 return
  7.         }
  8.         err = model.Db.Create(&lp).Error
  9.         if err != nil {
  10.                 http.Error(w, err.Error(), http.StatusInternalServerError)
  11.                 return
  12.         }
  13.         // 启动代理服务
  14.         go startProxy(lp)
  15.         libs.SuccessMsg(w, lp, "")
  16. }// GetLocalProxiesHandler 获取所有 LocalProxy 的 HTTP 处理函数func GetLocalProxiesHandler(w http.ResponseWriter, r *http.Request) {        // 获取查询参数 page 和 limit        pageStr := r.URL.Query().Get("page")        limitStr := r.URL.Query().Get("limit")        page, err := strconv.Atoi(pageStr)        if err != nil || page < 1 {                page = 1        }        limit, err := strconv.Atoi(limitStr)        if err != nil || limit < 1 {                limit = 10        }        // 定义相应结构体        type ProxyResponse struct {                Proxies []model.LocalProxy `json:"proxies"`                Total   int64              `json:"total"`        }        // 修改处理函数        proxies, total, err := model.GetLocalProxies(page, limit)        if err != nil {                http.Error(w, err.Error(), http.StatusInternalServerError)                return        }        libs.SuccessMsg(w, ProxyResponse{Proxies: proxies, Total: total}, "")}// GetLocalProxyHandler 获取单个 LocalProxy 的 HTTP 处理函数func GetLocalProxyHandler(w http.ResponseWriter, r *http.Request) {        idStr := r.URL.Query().Get("id")        id, err := strconv.Atoi(idStr)        if err != nil {                http.Error(w, "Invalid ID", http.StatusBadRequest)                return        }        var proxy model.LocalProxy        err = model.Db.First(&proxy, uint(id)).Error        if err != nil {                http.Error(w, err.Error(), http.StatusNotFound)                return        }        libs.SuccessMsg(w, proxy, "")}// UpdateLocalProxyHandler 更新 LocalProxy 的 HTTP 处理函数func UpdateLocalProxyHandler(w http.ResponseWriter, r *http.Request) {        var lp model.LocalProxy        err := json.NewDecoder(r.Body).Decode(&lp)        if err != nil {                http.Error(w, err.Error(), http.StatusBadRequest)                return        }        err = model.Db.Model(&model.LocalProxy{}).Where("id = ?", lp.ID).Updates(lp).Error        if err != nil {                http.Error(w, err.Error(), http.StatusInternalServerError)                return        }        // 制止旧的代理服务        stopProxy(lp.ID)        // 启动新的代理服务        go startProxy(lp)        libs.SuccessMsg(w, lp, "")}// DeleteLocalProxyHandler 删除 LocalProxy 的 HTTP 处理函数func DeleteLocalProxyHandler(w http.ResponseWriter, r *http.Request) {        idStr := r.URL.Query().Get("id")        id, err := strconv.Atoi(idStr)        if err != nil {                http.Error(w, "Invalid ID", http.StatusBadRequest)                return        }        // 制止代理服务        stopProxy(uint(id))        err = model.Db.Delete(&model.LocalProxy{}, uint(id)).Error        if err != nil {                http.Error(w, err.Error(), http.StatusInternalServerError)                return        }        libs.SuccessMsg(w, nil, "delete proxy success")}func HandlerSetProxyStatus(w http.ResponseWriter, r *http.Request) {        idStr := r.URL.Query().Get("id")        if idStr == "" {                libs.ErrorMsg(w, "id is empty")                return        }        id, err := strconv.Atoi(idStr)        if err != nil || id == 0 {                libs.ErrorMsg(w, "id is not number")                return        }        var proxy model.LocalProxy        if err := model.Db.First(&proxy, uint(id)).Error; err != nil {                libs.ErrorMsg(w, "proxy not found")                return        }        if err := model.Db.Model(&model.LocalProxy{}).Where("id = ?", proxy.ID).Update("status", !proxy.Status).Error; err != nil {                libs.ErrorMsg(w, "update proxy status failed")                return        }        if !proxy.Status {                startProxy(proxy)        } else {                stopProxy(proxy.ID)        }        libs.SuccessMsg(w, nil, "")}// 初始化代理处理函数func InitProxyHandlers() {        go InitFrpcServer()        proxies, _, err := model.GetLocalProxies(1, 1000) // 获取所有代理设置        if err != nil {                fmt.Println("Failed to get local proxies:", err)                return        }        for _, proxy := range proxies {                go startProxy(proxy)        }}// 启动代理服务func startProxy(proxy model.LocalProxy) {        switch proxy.ProxyType {        case "http":                go func(p model.LocalProxy) {                        httpProxyHandler(p)                }(proxy)        case "udp":                go func(p model.LocalProxy) {                        udpProxyHandler(p)                }(proxy)        case "file":                go func(p model.LocalProxy) {                        fileServerHandler(p)                }(proxy)        default:                fmt.Printf("Unknown proxy type: %s\n", proxy.ProxyType)        }}// 制止代理服务func stopProxy(id uint) {        if server, ok := proxyServers.Load(id); ok {                proxyServer, ok := server.(ProxyServer)                if !ok {                        fmt.Printf("Failed to load proxy server for ID %d\n", id)                        return                }                switch proxyServer.Type {                case "http":                        httpServer, ok := proxyServer.Server.(*http.Server)                        if ok {                                // 创建一个上下文,用于通报给 server.Shutdown(ctx)                                ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)                                defer cancel()                                if err := httpServer.Shutdown(ctx); err != nil {                                        fmt.Printf("Failed to shutdown HTTP server on port %s: %v\n", httpServer.Addr, err)                                } else {                                        fmt.Printf("Stopped HTTP server on port %s\n", httpServer.Addr)                                }                        }                case "udp":                        udpConn, ok := proxyServer.Server.(*net.UDPConn)                        if ok {                                udpConn.Close()                                fmt.Printf("Stopped UDP server on port %d\n", udpConn.LocalAddr().(*net.UDPAddr).Port)                        }                case "file":                        fileServer, ok := proxyServer.Server.(*FileServer)                        if ok {                                // 创建一个上下文,用于通报给 server.Shutdown(ctx)                                ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)                                defer cancel()                                if err := fileServer.Server.Shutdown(ctx); err != nil {                                        fmt.Printf("Failed to shutdown file server on port %d: %v\n", fileServer.Port, err)                                } else {                                        fmt.Printf("Stopped file server on port %d\n", fileServer.Port)                                }                        }                default:                        fmt.Printf("Unknown proxy type: %s\n", proxyServer.Type)                }                proxyServers.Delete(id)        }}// HTTP 代理处理函数func httpProxyHandler(proxy model.LocalProxy) {        remote, err := url.Parse(proxy.Domain)        if err != nil {                fmt.Printf("Failed to parse remote URL for port %d: %v\n", proxy.Port, err)                return        }        if remote.Scheme == "" {                fmt.Printf("Remote URL for port %d does not contain a scheme (http/https): %s\n", proxy.Port, proxy.Domain)                return        }        reverseProxy := httputil.NewSingleHostReverseProxy(remote)        // 设置请求头        // reverseProxy.Director = func(req *http.Request) {        //         req.Header.Add("X-Forwarded-For", req.RemoteAddr)        //         req.Header.Add("X-Real-IP", req.RemoteAddr)        //         req.Host = remote.Host        // }        // 启动 HTTP 服务器并监听指定端口        server := &http.Server{                Addr:    fmt.Sprintf(":%d", proxy.Port),                Handler: reverseProxy,        }        proxyServers.Store(proxy.ID, ProxyServer{Type: "http", Server: server})        fmt.Printf("Starting HTTP proxy on port %d and forwarding to %s:%d\n", proxy.Port, proxy.Domain, proxy.Port)        if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {                fmt.Printf("Failed to start HTTP proxy on port %d: %v\n", proxy.Port, err)        }}// UDP 代理处理函数func udpProxyHandler(proxy model.LocalProxy) {        localAddr, err := net.ResolveUDPAddr("udp", fmt.Sprintf(":%d", proxy.Port))        if err != nil {                fmt.Printf("Failed to resolve local UDP address for port %d: %v\n", proxy.Port, err)                return        }        remoteAddr, err := net.ResolveUDPAddr("udp", proxy.Domain)        if err != nil {                fmt.Printf("Failed to resolve remote UDP address for port %d: %v\n", proxy.Port, err)                return        }        conn, err := net.ListenUDP("udp", localAddr)        if err != nil {                fmt.Printf("Failed to listen on UDP port %d: %v\n", proxy.Port, err)                return        }        defer conn.Close()        proxyServers.Store(proxy.ID, ProxyServer{Type: "udp", Server: conn})        buffer := make([]byte, 1024)        for {                n, addr, err := conn.ReadFromUDP(buffer)                if err != nil {                        fmt.Printf("Failed to read from UDP: %v\n", err)                        continue                }                remoteConn, err := net.DialUDP("udp", nil, remoteAddr)                if err != nil {                        fmt.Printf("Failed to dial remote UDP: %v\n", err)                        continue                }                _, err = remoteConn.Write(buffer[:n])                if err != nil {                        fmt.Printf("Failed to write to remote UDP: %v\n", err)                        continue                }                // 读取远程服务器的相应并转发回客户端                n, err = remoteConn.Read(buffer)                if err != nil {                        fmt.Printf("Failed to read from remote UDP: %v\n", err)                        continue                }                _, err = conn.WriteToUDP(buffer[:n], addr)                if err != nil {                        fmt.Printf("Failed to write to client UDP: %v\n", err)                        continue                }        }}// 文件静态服务处理函数func fileServerHandler(proxy model.LocalProxy) {        // 创建文件服务器的 HTTP 处理函数        fileHandler := http.FileServer(http.Dir(proxy.Path))        // 启动 HTTP 服务器并监听指定端口        server := &http.Server{                Addr:    fmt.Sprintf(":%d", proxy.Port),                Handler: fileHandler,        }        proxyServers.Store(proxy.ID, ProxyServer{Type: "file", Server: &FileServer{Port: int(proxy.Port), Server: server}})        fmt.Printf("Starting file server on port %d serving files from %s\n", proxy.Port, proxy.Path)        if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {                fmt.Printf("Failed to start file server on port %d: %v\n", proxy.Port, err)        }}
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

圆咕噜咕噜

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表