Browse Source

获取城市树

icole 4 years ago
parent
commit
98a5389a15

+ 4 - 1
etc/transform.yaml

@@ -8,4 +8,7 @@ DataSource: root:gSRGZqb121TlYIbJy0@tcp(47.103.202.94:3306)/i2erp_erp1?charset=u
 Table:
 Cache:
   - Host: 47.103.219.158:7001
-    Pass: i2#333
+    Pass: i2#333
+Erp:
+  AutoRefresh: true
+  AuthServer: 47.103.153.70:31134

+ 4 - 0
internal/config/config.go

@@ -10,4 +10,8 @@ type Config struct {
 	DataSource string
 	Table      string
 	Cache      cache.CacheConf
+	Erp        struct {
+		AutoRefresh bool
+		AuthServer  string
+	}
 }

+ 34 - 0
internal/logic/parsetokenlogic.go

@@ -0,0 +1,34 @@
+package logic
+
+import (
+	"context"
+
+	"git.i2edu.net/i2/i2-bill-erp/internal/svc"
+	"git.i2edu.net/i2/i2-bill-erp/internal/utils"
+	"git.i2edu.net/i2/i2-bill-erp/transform"
+
+	"git.i2edu.net/i2/go-zero/core/logx"
+)
+
+type ParseTokenLogic struct {
+	ctx    context.Context
+	svcCtx *svc.ServiceContext
+	logx.Logger
+}
+
+func NewParseTokenLogic(ctx context.Context, svcCtx *svc.ServiceContext) *ParseTokenLogic {
+	return &ParseTokenLogic{
+		ctx:    ctx,
+		svcCtx: svcCtx,
+		Logger: logx.WithContext(ctx),
+	}
+}
+
+func (l *ParseTokenLogic) ParseToken(in *transform.TokenRequest) (*transform.TokenResponse, error) {
+	erpToken := utils.GetGlobalTokenStore().Get(in.Token)
+	tr := transform.TokenResponse{}
+	if erpToken != nil {
+		tr.UserId = erpToken.UserId
+	}
+	return &tr, nil
+}

+ 6 - 1
internal/server/transformserver.go

@@ -26,7 +26,12 @@ func (s *TransformServer) GetUser(ctx context.Context, in *transform.UserRequest
 	return l.GetUser(in)
 }
 
+func (s *TransformServer) ParseToken(ctx context.Context, in *transform.TokenRequest) (*transform.TokenResponse, error) {
+	l := logic.NewParseTokenLogic(ctx, s.svcCtx)
+	return l.ParseToken(in)
+}
+
 func (s *TransformServer) GetErpCityTree(ctx context.Context, in *transform.Empty) (*transform.TreeNodes, error) {
 	l := logic.NewGetErpCityTreeLogic(ctx, s.svcCtx)
-	return l.GetErpCityTree()
+	return l.GetErpCityTree(in)
 }

+ 633 - 0
internal/utils/token.go

@@ -0,0 +1,633 @@
+// ERP 代码, light的逻辑, 废弃!!!!!
+// ERP 代码, light的逻辑, 废弃!!!!!
+
+package utils
+
+import (
+	"bytes"
+	"crypto/md5"
+	"encoding/binary"
+	"encoding/hex"
+	"errors"
+	"fmt"
+	"log"
+	"net"
+	"strconv"
+	"strings"
+	"sync"
+	"time"
+
+	"git.i2edu.net/i2/go-zero/core/logx"
+	"git.i2edu.net/i2/i2-bill-erp/internal/config"
+)
+
+type Token struct {
+	Lock         *sync.RWMutex
+	Result       int                    `json:"-"`
+	UserId       string                 `json:"user_id"`
+	AccessToken  string                 `json:"access_token"`
+	RefreshToken string                 `json:"refresh_token"`
+	LoginID      string                 `json:"login_id"`
+	TimeStamp    uint64                 `json:"time_stamp"`
+	ServerIp     string                 `json:"server_ip"`
+	Domain       string                 `json:"domain"`
+	Password     string                 `json:"-"`
+	Map          map[string]interface{} `json:"map"`
+}
+
+type TokenStore struct {
+	name   string
+	lock   *sync.RWMutex
+	tokens map[string]*Token
+}
+
+type IAuth interface {
+	Init()
+}
+
+var globalTokenStore *TokenStore = nil
+
+func init() {
+	iauthMap = make(map[string]IAuth)
+	globalTokenStore = &TokenStore{name: "sso", lock: new(sync.RWMutex), tokens: make(map[string]*Token)}
+	go globalTokenStore.startTokenCheckProcess()
+
+	lightAuth := &LightAuth{}
+	RegisterAuth("qianqiusoft.com", lightAuth)
+}
+
+type LightAuth struct {
+}
+
+func (la *LightAuth) Init() {
+
+}
+
+var iauthMap map[string]IAuth
+
+/**
+ * @brief: 注册自定登录对象, 会调用Init进行初始化
+ * @param1 t: key
+ * @param2 a: 认证对象
+ * @return none
+ */
+func RegisterAuth(t string, a IAuth) {
+	if t == "" || a == nil {
+		return
+	}
+
+	var ok bool
+	if _, ok = iauthMap[t]; ok {
+		return
+	} else {
+		a.Init()
+		iauthMap[t] = a
+	}
+}
+
+func GetGlobalTokenStore() *TokenStore {
+	return globalTokenStore
+}
+
+func (t *TokenStore) Get(key string) *Token {
+	t.lock.RLock()
+	defer t.lock.RUnlock()
+	if val, ok := t.tokens[key]; ok {
+		//log.Println(key, "获取Token:", val.AccessToken, val.RefreshToken, val.LoginID)
+		return val
+	}
+	return nil
+}
+
+func (t *TokenStore) Set(key string, v *Token) {
+	t.lock.Lock()
+	defer t.lock.Unlock()
+	if val, ok := t.tokens[key]; !ok {
+		t.tokens[key] = v
+	} else if val != v {
+		t.tokens[key] = v
+	}
+}
+
+func (t *TokenStore) Remove(key string) {
+	t.lock.Lock()
+	defer t.lock.Unlock()
+	delete(t.tokens, key)
+	log.Println(key, "删除Key")
+}
+
+func (t *TokenStore) Refresh(key string) {
+	t.lock.Lock()
+	defer t.lock.Unlock()
+
+	val, ok := t.tokens[key]
+	if ok {
+		val.TimeStamp = uint64(time.Now().UnixNano())
+	}
+}
+
+func (t *TokenStore) startTokenCheckProcess() {
+}
+
+func Validate(accessToken, loginId string, domain string) (*Token, error) {
+	token := globalTokenStore.Get(loginId + domain)
+	if token != nil {
+		if strings.EqualFold(token.AccessToken, accessToken) {
+			logx.Info("get the token ", accessToken, " of id ", loginId+domain)
+			globalTokenStore.Refresh(loginId + domain)
+			return token, nil
+		} else {
+			logx.Error(token.AccessToken, "is not equal to", accessToken)
+			return token, errors.New(token.AccessToken + " is not equal to " + accessToken)
+		}
+	} else {
+		logx.Error("can not get the token of", loginId+domain)
+		return token, errors.New("can not get the token of " + loginId + domain)
+	}
+
+}
+
+func TokenValidate(token string) (*Token, error) {
+	user := globalTokenStore.Get(token)
+
+	if strings.EqualFold(user.AccessToken, token) {
+		logx.Info("get the token ", token, " of id ")
+		globalTokenStore.Refresh(token)
+		return user, nil
+	} else {
+		logx.Error(user.AccessToken, "is not equal to", token)
+		return user, errors.New(user.AccessToken + " is not equal to " + token)
+	}
+
+}
+
+const (
+	__KEY = "Light#dauth-@*I2"
+
+	CMD_NEW           = "new"
+	CMD_REMOVE        = "remove"
+	CMD_PINGPONG      = "pg"
+	CMD_PINGPONG_RESP = "pg_resp"
+)
+
+var TCPClient *TcpClient
+
+type authPackage struct {
+	Cmd      string
+	TokenStr string
+	Content  []byte
+}
+
+func (ap *authPackage) toBytes() []byte {
+	buf := bytes.NewBuffer([]byte{})
+
+	b := []byte(ap.Cmd)
+	buf.Write(uint32ToBytes(len(b)))
+	buf.Write(b)
+
+	buf.Write(uint32ToBytes(len(ap.Content)))
+	buf.Write(ap.Content)
+
+	return buf.Bytes()
+}
+
+type TcpClient struct {
+	conn     net.Conn // 连接
+	c        *config.Config
+	pchan    chan *authPackage // 包chan
+	done     chan bool         // 是否done
+	exited   bool              // 退出
+	verified bool              // 验证是否
+}
+
+// 创建client
+func NewTcpClient(cfg *config.Config) *TcpClient {
+	c := &TcpClient{
+		c:        cfg,
+		pchan:    make(chan *authPackage, 100),
+		done:     make(chan bool),
+		exited:   false,
+		verified: false,
+	}
+	return c
+}
+
+// 启动
+func (c *TcpClient) Start() {
+	go func() {
+		defer func() {
+			if p := recover(); p != nil {
+				fmt.Println("ecover", p)
+			}
+			if c.conn != nil {
+				c.conn.Close()
+			}
+			c.restart()
+		}()
+
+		var err error = nil
+		address := c.c.Erp.AuthServer
+		// fmt.Println("auth client start, dial address is", address)
+		c.conn, err = net.Dial("tcp", address)
+		if err != nil {
+			fmt.Println("Error dialing erp client", err.Error())
+			return
+		}
+		fmt.Println("发送验证")
+		sendVerify(c.conn) // 发送验证,不需要读取返回值,如果验证错误立刻关掉
+
+		fmt.Println("读取验证结果")
+		vresp, err := readString(c.conn)
+		if err != nil {
+			fmt.Println("Error dialing", err.Error())
+			return
+		}
+		if vresp != "ok" {
+			// 验证失败
+			fmt.Println("verify is not ok", vresp)
+			return
+		}
+		fmt.Println("验证成功")
+		c.verified = true
+
+		// send
+		go func() {
+			for {
+				select {
+				case data := <-c.pchan:
+					logx.Info("写入数据")
+					c.conn.SetWriteDeadline(time.Now().Add(time.Second * 2))
+					_, err := c.conn.Write(data.toBytes())
+					if err != nil {
+						fmt.Println("写入内容错误", err.Error())
+						return
+					}
+				case <-c.done:
+					logx.Info("发送数据done退出")
+					return
+				}
+			}
+		}()
+
+		// receive
+		for {
+			cmd, err := readString(c.conn) // 读取命令
+			if err != nil {
+				c.done <- true
+				fmt.Println("读取命令错误", err.Error())
+				break
+			}
+			if cmd == CMD_NEW {
+				err = c.newHandler()
+			} else if cmd == CMD_REMOVE {
+				err = c.removeHandler()
+			} else if cmd == CMD_PINGPONG_RESP {
+
+			} else {
+				fmt.Println("未知cmd", cmd)
+				continue
+			}
+			if err != nil {
+				c.done <- true
+				fmt.Println("处理错误", err.Error())
+				break
+			}
+		}
+	}()
+}
+
+// 停止
+func (c *TcpClient) Stop() {
+	c.exited = true
+	c.conn.Close()
+}
+
+// 检测
+func (c *TcpClient) restart() {
+	if c.exited {
+		// 已退出则不重启
+		return
+	}
+
+	go func() {
+		c.verified = false
+		c.done = make(chan bool)
+		c.pchan = make(chan *authPackage, 100)
+		c.exited = false
+		time.Sleep(3 * time.Second)
+		c.Start()
+	}()
+}
+
+// 发送bytes
+func (c *TcpClient) Send(cmd string, bytess []byte) {
+	if !c.verified {
+		fmt.Println("未认证")
+		return
+	}
+
+	logx.Info("发送指令1", cmd)
+	c.pchan <- &authPackage{
+		Cmd:     cmd,
+		Content: bytess,
+	}
+	logx.Info("发送指令2", cmd)
+}
+
+// 发送token
+func (c *TcpClient) SendToken(token *Token) {
+	// logx.Info("发送新建token")
+	bytess := tokenToBytes(token)
+	c.Send(CMD_NEW, bytess)
+}
+
+// 处理创建
+func (c *TcpClient) newHandler() error {
+	// fmt.Println("处理新建")
+	bytess, err := readBytes(c.conn)
+	if err != nil {
+		fmt.Println("读取token内容错误", err.Error())
+		return err
+	}
+	// 新建
+	token, err := bytesToToken(bytess)
+	if err != nil {
+		logx.Info("bytesToToken 错误", err.Error())
+		return err
+	}
+	GetGlobalTokenStore().Set(token.AccessToken, token)
+	return nil
+}
+
+// 处理删除
+func (c *TcpClient) removeHandler() error {
+	fmt.Println("处理删除")
+	bytess, err := readBytes(c.conn)
+	if err != nil {
+		fmt.Println("读取token内容错误", err.Error())
+		return err
+	}
+	// 移除,此时bytess为tokenstring
+	GetGlobalTokenStore().Remove(string(bytess))
+	return nil
+}
+
+// 读取字符串
+func readString(conn net.Conn) (string, error) {
+	// 读长度
+	size, err := readUInt32(conn)
+	if err != nil {
+		fmt.Println("读取长度失败,", err.Error())
+		return "", err
+	}
+
+	// 读字符串
+	b := make([]byte, size)
+	n, err := conn.Read(b)
+	if n != int(size) {
+		return "", errors.New("读取长度不是" + strconv.Itoa(int(size)))
+	}
+	return string(b), nil
+}
+
+// 写入字符串
+func writeString(conn net.Conn, str string) error {
+	if str == "" {
+		return errors.New("字符串为空")
+	}
+
+	bytess := []byte(str)
+	size := len(bytess)
+
+	// 发送长度
+	err := writeUInt32(conn, uint32(size))
+	if err != nil {
+		fmt.Println("发送长度失败,", err.Error())
+		return err
+	}
+
+	// 发送内容
+	n, err := conn.Write(bytess)
+	if err != nil {
+		fmt.Println("发送内容失败,", err.Error())
+		return err
+	}
+	if n != size {
+		return errors.New("发送长度不是" + strconv.Itoa(int(size)))
+	}
+
+	return nil
+}
+
+// 读取bytes
+func readBytes(conn net.Conn) ([]byte, error) {
+	// 读长度
+	size, err := readUInt32(conn)
+	if err != nil {
+		fmt.Println("读取长度失败,", err.Error())
+		return nil, err
+	}
+
+	// 读字符串
+	b := make([]byte, size)
+	n, err := conn.Read(b)
+	if n != int(size) {
+		return nil, errors.New("读取长度不是" + strconv.Itoa(int(size)))
+	}
+	return b, nil
+}
+
+// 读取uint64
+func readUInt32(conn net.Conn) (uint32, error) {
+	b := make([]byte, 4)
+	n, err := conn.Read(b)
+	if err != nil {
+		fmt.Println("读取长度失败,", err.Error())
+		return 0, err
+	}
+	if n != 4 {
+		return 0, errors.New("读取长度不是4")
+	}
+	size := binary.BigEndian.Uint32(b)
+
+	return size, nil
+}
+
+// 写入长度
+func writeUInt32(conn net.Conn, v uint32) error {
+	// 发送长度
+	b := make([]byte, 4)
+	binary.BigEndian.PutUint32(b, v)
+	n, err := conn.Write(b)
+	if err != nil {
+		fmt.Println("发送长度失败,", err.Error())
+		return err
+	}
+	if n != 4 {
+		return errors.New("发送长度不是4")
+	}
+
+	return nil
+}
+
+// 写入长度
+func writeUInt64(conn net.Conn, v uint64) error {
+	// 发送长度
+	b := make([]byte, 8)
+	binary.BigEndian.PutUint64(b, v)
+	n, err := conn.Write(b)
+	if err != nil {
+		fmt.Println("发送长度失败,", err.Error())
+		return err
+	}
+	if n != 4 {
+		return errors.New("发送长度不是4")
+	}
+
+	return nil
+}
+
+// 读取uint64
+func readStringByBytes(bytess []byte) (string, int, error) {
+	size := binary.BigEndian.Uint32(bytess)
+	return string(bytess[4 : 4+size]), int(size), nil
+}
+
+// int转bytes
+func uint32ToBytes(v int) []byte {
+	b := make([]byte, 4)
+	binary.BigEndian.PutUint32(b, uint32(v))
+
+	return b
+}
+
+// int转bytes
+func uint64ToBytes(v int) []byte {
+	b := make([]byte, 8)
+	binary.BigEndian.PutUint32(b, uint32(v))
+
+	return b
+}
+
+// 转token
+func bytesToToken(content []byte) (*Token, error) {
+	token := &Token{Lock: new(sync.RWMutex)}
+	var index int = 0
+	var size int
+	var err error = nil
+	// fmt.Println("读取userid")
+	token.UserId, size, err = readStringByBytes(content)
+	if err != nil {
+		fmt.Println("读取userid错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	// fmt.Println("读取AccessToken")
+	token.AccessToken, size, err = readStringByBytes(content[index:])
+	if err != nil {
+		fmt.Println("读取AccessToken错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	// fmt.Println("读取RefreshToken")
+	token.RefreshToken, size, err = readStringByBytes(content[index:])
+	if err != nil {
+		fmt.Println("读取RefreshToken错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	// fmt.Println("读取LoginID")
+	token.LoginID, size, err = readStringByBytes(content[index:])
+	if err != nil {
+		fmt.Println("读取LoginID错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	// fmt.Println("读取timestamp")
+	token.TimeStamp = binary.BigEndian.Uint64(content[index:])
+	index += 8
+
+	// fmt.Println("读取ServerIp")
+	token.ServerIp, size, err = readStringByBytes(content[index:])
+	if err != nil {
+		fmt.Println("读取ServerIp错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	// fmt.Println("读取Domain")
+	token.Domain, size, err = readStringByBytes(content[index:])
+	if err != nil {
+		fmt.Println("读取Domain错误")
+		return nil, err
+	}
+	index += 4 + size
+
+	return token, nil
+}
+
+// 转bytes,包括token开头
+func tokenToBytes(token *Token) []byte {
+	buf := bytes.NewBuffer([]byte{})
+
+	t := []byte(token.UserId)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	t = []byte(token.AccessToken)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	t = []byte(token.RefreshToken)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	t = []byte(token.LoginID)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	buf.Write(uint64ToBytes(int(token.TimeStamp)))
+
+	fmt.Println(token.ServerIp)
+	t = []byte(token.ServerIp)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	fmt.Println(token.Domain)
+	t = []byte(token.Domain)
+	buf.Write(uint32ToBytes(len(t)))
+	buf.Write(t)
+
+	bytess := buf.Bytes()
+	buf = bytes.NewBuffer([]byte{}) // 这里用reset是错误的
+
+	tokenstrbytes := []byte(token.AccessToken)
+	buf.Write(uint32ToBytes(len(tokenstrbytes)))
+	buf.Write(tokenstrbytes)
+	buf.Write(uint32ToBytes(len(bytess)))
+	buf.Write(bytess)
+
+	return buf.Bytes()
+}
+
+// 发送验证
+func sendVerify(conn net.Conn) {
+	timestamp := time.Now().UnixNano()
+	timestampStr := strconv.Itoa(int(timestamp))
+	seed := timestampStr + __KEY
+	hashVal := hash(seed)
+
+	writeUInt64(conn, uint64(timestamp))
+	writeString(conn, hashVal)
+}
+
+// md5 哈希
+func hash(str string) string {
+	h := md5.New()
+	h.Write([]byte(str))
+	return hex.EncodeToString(h.Sum(nil))
+}

+ 6 - 0
transform.go

@@ -6,6 +6,7 @@ import (
 	"git.i2edu.net/i2/i2-bill-erp/internal/config"
 	"git.i2edu.net/i2/i2-bill-erp/internal/server"
 	"git.i2edu.net/i2/i2-bill-erp/internal/svc"
+	"git.i2edu.net/i2/i2-bill-erp/internal/utils"
 	"git.i2edu.net/i2/i2-bill-erp/transform"
 
 	"git.i2edu.net/i2/go-zero/core/conf"
@@ -20,6 +21,11 @@ func main() {
 
 	var c config.Config
 	conf.MustLoad(*configFile, &c)
+
+	// erp client
+	erpClient := utils.NewTcpClient(&c)
+	erpClient.Start()
+
 	ctx := svc.NewServiceContext(c)
 	srv := server.NewTransformServer(ctx)
 

+ 9 - 1
transform.proto

@@ -2,7 +2,6 @@ syntax = "proto3";
 
 package transform;
 
-
 message UserRequest {
   string id = 1;
   string phone = 2;
@@ -19,6 +18,14 @@ message UserResponse {
   int64 gender          = 8;
 }
 
+message TokenRequest {
+  string token = 1;
+}
+
+message TokenResponse {
+  string UserId = 1;
+}
+
 message Empty {}
 
 message TreeNodes{
@@ -42,5 +49,6 @@ message Tag{
 
 service Transform {
   rpc GetUser(UserRequest) returns(UserResponse);
+  rpc ParseToken(TokenRequest) returns(TokenResponse);
   rpc GetErpCityTree(Empty) returns(TreeNodes);
 }

+ 219 - 52
transform/transform.pb.go

@@ -187,6 +187,100 @@ func (x *UserResponse) GetGender() int64 {
 	return 0
 }
 
+type TokenRequest struct {
+	state         protoimpl.MessageState
+	sizeCache     protoimpl.SizeCache
+	unknownFields protoimpl.UnknownFields
+
+	Token string `protobuf:"bytes,1,opt,name=token,proto3" json:"token,omitempty"`
+}
+
+func (x *TokenRequest) Reset() {
+	*x = TokenRequest{}
+	if protoimpl.UnsafeEnabled {
+		mi := &file_transform_proto_msgTypes[2]
+		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
+		ms.StoreMessageInfo(mi)
+	}
+}
+
+func (x *TokenRequest) String() string {
+	return protoimpl.X.MessageStringOf(x)
+}
+
+func (*TokenRequest) ProtoMessage() {}
+
+func (x *TokenRequest) ProtoReflect() protoreflect.Message {
+	mi := &file_transform_proto_msgTypes[2]
+	if protoimpl.UnsafeEnabled && x != nil {
+		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
+		if ms.LoadMessageInfo() == nil {
+			ms.StoreMessageInfo(mi)
+		}
+		return ms
+	}
+	return mi.MessageOf(x)
+}
+
+// Deprecated: Use TokenRequest.ProtoReflect.Descriptor instead.
+func (*TokenRequest) Descriptor() ([]byte, []int) {
+	return file_transform_proto_rawDescGZIP(), []int{2}
+}
+
+func (x *TokenRequest) GetToken() string {
+	if x != nil {
+		return x.Token
+	}
+	return ""
+}
+
+type TokenResponse struct {
+	state         protoimpl.MessageState
+	sizeCache     protoimpl.SizeCache
+	unknownFields protoimpl.UnknownFields
+
+	UserId string `protobuf:"bytes,1,opt,name=UserId,proto3" json:"UserId,omitempty"`
+}
+
+func (x *TokenResponse) Reset() {
+	*x = TokenResponse{}
+	if protoimpl.UnsafeEnabled {
+		mi := &file_transform_proto_msgTypes[3]
+		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
+		ms.StoreMessageInfo(mi)
+	}
+}
+
+func (x *TokenResponse) String() string {
+	return protoimpl.X.MessageStringOf(x)
+}
+
+func (*TokenResponse) ProtoMessage() {}
+
+func (x *TokenResponse) ProtoReflect() protoreflect.Message {
+	mi := &file_transform_proto_msgTypes[3]
+	if protoimpl.UnsafeEnabled && x != nil {
+		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
+		if ms.LoadMessageInfo() == nil {
+			ms.StoreMessageInfo(mi)
+		}
+		return ms
+	}
+	return mi.MessageOf(x)
+}
+
+// Deprecated: Use TokenResponse.ProtoReflect.Descriptor instead.
+func (*TokenResponse) Descriptor() ([]byte, []int) {
+	return file_transform_proto_rawDescGZIP(), []int{3}
+}
+
+func (x *TokenResponse) GetUserId() string {
+	if x != nil {
+		return x.UserId
+	}
+	return ""
+}
+
 type Empty struct {
 	state         protoimpl.MessageState
 	sizeCache     protoimpl.SizeCache
@@ -196,7 +290,7 @@ type Empty struct {
 func (x *Empty) Reset() {
 	*x = Empty{}
 	if protoimpl.UnsafeEnabled {
-		mi := &file_transform_proto_msgTypes[2]
+		mi := &file_transform_proto_msgTypes[4]
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		ms.StoreMessageInfo(mi)
 	}
@@ -209,7 +303,7 @@ func (x *Empty) String() string {
 func (*Empty) ProtoMessage() {}
 
 func (x *Empty) ProtoReflect() protoreflect.Message {
-	mi := &file_transform_proto_msgTypes[2]
+	mi := &file_transform_proto_msgTypes[4]
 	if protoimpl.UnsafeEnabled && x != nil {
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		if ms.LoadMessageInfo() == nil {
@@ -222,7 +316,7 @@ func (x *Empty) ProtoReflect() protoreflect.Message {
 
 // Deprecated: Use Empty.ProtoReflect.Descriptor instead.
 func (*Empty) Descriptor() ([]byte, []int) {
-	return file_transform_proto_rawDescGZIP(), []int{2}
+	return file_transform_proto_rawDescGZIP(), []int{4}
 }
 
 type TreeNodes struct {
@@ -236,7 +330,7 @@ type TreeNodes struct {
 func (x *TreeNodes) Reset() {
 	*x = TreeNodes{}
 	if protoimpl.UnsafeEnabled {
-		mi := &file_transform_proto_msgTypes[3]
+		mi := &file_transform_proto_msgTypes[5]
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		ms.StoreMessageInfo(mi)
 	}
@@ -249,7 +343,7 @@ func (x *TreeNodes) String() string {
 func (*TreeNodes) ProtoMessage() {}
 
 func (x *TreeNodes) ProtoReflect() protoreflect.Message {
-	mi := &file_transform_proto_msgTypes[3]
+	mi := &file_transform_proto_msgTypes[5]
 	if protoimpl.UnsafeEnabled && x != nil {
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		if ms.LoadMessageInfo() == nil {
@@ -262,7 +356,7 @@ func (x *TreeNodes) ProtoReflect() protoreflect.Message {
 
 // Deprecated: Use TreeNodes.ProtoReflect.Descriptor instead.
 func (*TreeNodes) Descriptor() ([]byte, []int) {
-	return file_transform_proto_rawDescGZIP(), []int{3}
+	return file_transform_proto_rawDescGZIP(), []int{5}
 }
 
 func (x *TreeNodes) GetNodes() []*TreeNode {
@@ -287,7 +381,7 @@ type TreeNode struct {
 func (x *TreeNode) Reset() {
 	*x = TreeNode{}
 	if protoimpl.UnsafeEnabled {
-		mi := &file_transform_proto_msgTypes[4]
+		mi := &file_transform_proto_msgTypes[6]
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		ms.StoreMessageInfo(mi)
 	}
@@ -300,7 +394,7 @@ func (x *TreeNode) String() string {
 func (*TreeNode) ProtoMessage() {}
 
 func (x *TreeNode) ProtoReflect() protoreflect.Message {
-	mi := &file_transform_proto_msgTypes[4]
+	mi := &file_transform_proto_msgTypes[6]
 	if protoimpl.UnsafeEnabled && x != nil {
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		if ms.LoadMessageInfo() == nil {
@@ -313,7 +407,7 @@ func (x *TreeNode) ProtoReflect() protoreflect.Message {
 
 // Deprecated: Use TreeNode.ProtoReflect.Descriptor instead.
 func (*TreeNode) Descriptor() ([]byte, []int) {
-	return file_transform_proto_rawDescGZIP(), []int{4}
+	return file_transform_proto_rawDescGZIP(), []int{6}
 }
 
 func (x *TreeNode) GetId() int64 {
@@ -363,7 +457,7 @@ type Tag struct {
 func (x *Tag) Reset() {
 	*x = Tag{}
 	if protoimpl.UnsafeEnabled {
-		mi := &file_transform_proto_msgTypes[5]
+		mi := &file_transform_proto_msgTypes[7]
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		ms.StoreMessageInfo(mi)
 	}
@@ -376,7 +470,7 @@ func (x *Tag) String() string {
 func (*Tag) ProtoMessage() {}
 
 func (x *Tag) ProtoReflect() protoreflect.Message {
-	mi := &file_transform_proto_msgTypes[5]
+	mi := &file_transform_proto_msgTypes[7]
 	if protoimpl.UnsafeEnabled && x != nil {
 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x))
 		if ms.LoadMessageInfo() == nil {
@@ -389,7 +483,7 @@ func (x *Tag) ProtoReflect() protoreflect.Message {
 
 // Deprecated: Use Tag.ProtoReflect.Descriptor instead.
 func (*Tag) Descriptor() ([]byte, []int) {
-	return file_transform_proto_rawDescGZIP(), []int{5}
+	return file_transform_proto_rawDescGZIP(), []int{7}
 }
 
 func (x *Tag) GetId() int64 {
@@ -427,27 +521,36 @@ var file_transform_proto_rawDesc = []byte{
 	0x4e, 0x61, 0x6d, 0x65, 0x12, 0x1a, 0x0a, 0x08, 0x63, 0x69, 0x74, 0x79, 0x4e, 0x61, 0x6d, 0x65,
 	0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x63, 0x69, 0x74, 0x79, 0x4e, 0x61, 0x6d, 0x65,
 	0x12, 0x16, 0x0a, 0x06, 0x67, 0x65, 0x6e, 0x64, 0x65, 0x72, 0x18, 0x08, 0x20, 0x01, 0x28, 0x03,
-	0x52, 0x06, 0x67, 0x65, 0x6e, 0x64, 0x65, 0x72, 0x22, 0x07, 0x0a, 0x05, 0x45, 0x6d, 0x70, 0x74,
-	0x79, 0x22, 0x36, 0x0a, 0x09, 0x54, 0x72, 0x65, 0x65, 0x4e, 0x6f, 0x64, 0x65, 0x73, 0x12, 0x29,
-	0x0a, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x13, 0x2e,
-	0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x72, 0x65, 0x65, 0x4e, 0x6f,
-	0x64, 0x65, 0x52, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x22, 0x93, 0x01, 0x0a, 0x08, 0x54, 0x72,
-	0x65, 0x65, 0x4e, 0x6f, 0x64, 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01,
-	0x28, 0x03, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x74, 0x65, 0x78, 0x74, 0x18, 0x02,
-	0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x74, 0x65, 0x78, 0x74, 0x12, 0x16, 0x0a, 0x06, 0x70, 0x61,
-	0x72, 0x65, 0x6e, 0x74, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x06, 0x70, 0x61, 0x72, 0x65,
-	0x6e, 0x74, 0x12, 0x20, 0x0a, 0x03, 0x74, 0x61, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32,
-	0x0e, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x61, 0x67, 0x52,
-	0x03, 0x74, 0x61, 0x67, 0x12, 0x29, 0x0a, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x18, 0x05, 0x20,
-	0x03, 0x28, 0x0b, 0x32, 0x13, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e,
-	0x54, 0x72, 0x65, 0x65, 0x4e, 0x6f, 0x64, 0x65, 0x52, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x22,
-	0x29, 0x0a, 0x03, 0x54, 0x61, 0x67, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01,
-	0x28, 0x03, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x02,
-	0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x32, 0x81, 0x01, 0x0a, 0x09, 0x54,
-	0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x12, 0x3a, 0x0a, 0x07, 0x47, 0x65, 0x74, 0x55,
-	0x73, 0x65, 0x72, 0x12, 0x16, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e,
-	0x55, 0x73, 0x65, 0x72, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x17, 0x2e, 0x74, 0x72,
-	0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x55, 0x73, 0x65, 0x72, 0x52, 0x65, 0x73, 0x70,
+	0x52, 0x06, 0x67, 0x65, 0x6e, 0x64, 0x65, 0x72, 0x22, 0x24, 0x0a, 0x0c, 0x54, 0x6f, 0x6b, 0x65,
+	0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x14, 0x0a, 0x05, 0x74, 0x6f, 0x6b, 0x65,
+	0x6e, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x74, 0x6f, 0x6b, 0x65, 0x6e, 0x22, 0x27,
+	0x0a, 0x0d, 0x54, 0x6f, 0x6b, 0x65, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12,
+	0x16, 0x0a, 0x06, 0x55, 0x73, 0x65, 0x72, 0x49, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52,
+	0x06, 0x55, 0x73, 0x65, 0x72, 0x49, 0x64, 0x22, 0x07, 0x0a, 0x05, 0x45, 0x6d, 0x70, 0x74, 0x79,
+	0x22, 0x36, 0x0a, 0x09, 0x54, 0x72, 0x65, 0x65, 0x4e, 0x6f, 0x64, 0x65, 0x73, 0x12, 0x29, 0x0a,
+	0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x13, 0x2e, 0x74,
+	0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x72, 0x65, 0x65, 0x4e, 0x6f, 0x64,
+	0x65, 0x52, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x22, 0x93, 0x01, 0x0a, 0x08, 0x54, 0x72, 0x65,
+	0x65, 0x4e, 0x6f, 0x64, 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28,
+	0x03, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x74, 0x65, 0x78, 0x74, 0x18, 0x02, 0x20,
+	0x01, 0x28, 0x09, 0x52, 0x04, 0x74, 0x65, 0x78, 0x74, 0x12, 0x16, 0x0a, 0x06, 0x70, 0x61, 0x72,
+	0x65, 0x6e, 0x74, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x06, 0x70, 0x61, 0x72, 0x65, 0x6e,
+	0x74, 0x12, 0x20, 0x0a, 0x03, 0x74, 0x61, 0x67, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x0e,
+	0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x61, 0x67, 0x52, 0x03,
+	0x74, 0x61, 0x67, 0x12, 0x29, 0x0a, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x18, 0x05, 0x20, 0x03,
+	0x28, 0x0b, 0x32, 0x13, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54,
+	0x72, 0x65, 0x65, 0x4e, 0x6f, 0x64, 0x65, 0x52, 0x05, 0x6e, 0x6f, 0x64, 0x65, 0x73, 0x22, 0x29,
+	0x0a, 0x03, 0x54, 0x61, 0x67, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28,
+	0x03, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x02, 0x20,
+	0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x32, 0xc2, 0x01, 0x0a, 0x09, 0x54, 0x72,
+	0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x12, 0x3a, 0x0a, 0x07, 0x47, 0x65, 0x74, 0x55, 0x73,
+	0x65, 0x72, 0x12, 0x16, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x55,
+	0x73, 0x65, 0x72, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x17, 0x2e, 0x74, 0x72, 0x61,
+	0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x55, 0x73, 0x65, 0x72, 0x52, 0x65, 0x73, 0x70, 0x6f,
+	0x6e, 0x73, 0x65, 0x12, 0x3f, 0x0a, 0x0a, 0x50, 0x61, 0x72, 0x73, 0x65, 0x54, 0x6f, 0x6b, 0x65,
+	0x6e, 0x12, 0x17, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x6f,
+	0x6b, 0x65, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x18, 0x2e, 0x74, 0x72, 0x61,
+	0x6e, 0x73, 0x66, 0x6f, 0x72, 0x6d, 0x2e, 0x54, 0x6f, 0x6b, 0x65, 0x6e, 0x52, 0x65, 0x73, 0x70,
 	0x6f, 0x6e, 0x73, 0x65, 0x12, 0x38, 0x0a, 0x0e, 0x47, 0x65, 0x74, 0x45, 0x72, 0x70, 0x43, 0x69,
 	0x74, 0x79, 0x54, 0x72, 0x65, 0x65, 0x12, 0x10, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73, 0x66, 0x6f,
 	0x72, 0x6d, 0x2e, 0x45, 0x6d, 0x70, 0x74, 0x79, 0x1a, 0x14, 0x2e, 0x74, 0x72, 0x61, 0x6e, 0x73,
@@ -467,25 +570,29 @@ func file_transform_proto_rawDescGZIP() []byte {
 	return file_transform_proto_rawDescData
 }
 
-var file_transform_proto_msgTypes = make([]protoimpl.MessageInfo, 6)
+var file_transform_proto_msgTypes = make([]protoimpl.MessageInfo, 8)
 var file_transform_proto_goTypes = []interface{}{
-	(*UserRequest)(nil),  // 0: transform.UserRequest
-	(*UserResponse)(nil), // 1: transform.UserResponse
-	(*Empty)(nil),        // 2: transform.Empty
-	(*TreeNodes)(nil),    // 3: transform.TreeNodes
-	(*TreeNode)(nil),     // 4: transform.TreeNode
-	(*Tag)(nil),          // 5: transform.Tag
+	(*UserRequest)(nil),   // 0: transform.UserRequest
+	(*UserResponse)(nil),  // 1: transform.UserResponse
+	(*TokenRequest)(nil),  // 2: transform.TokenRequest
+	(*TokenResponse)(nil), // 3: transform.TokenResponse
+	(*Empty)(nil),         // 4: transform.Empty
+	(*TreeNodes)(nil),     // 5: transform.TreeNodes
+	(*TreeNode)(nil),      // 6: transform.TreeNode
+	(*Tag)(nil),           // 7: transform.Tag
 }
 var file_transform_proto_depIdxs = []int32{
-	4, // 0: transform.TreeNodes.nodes:type_name -> transform.TreeNode
-	5, // 1: transform.TreeNode.tag:type_name -> transform.Tag
-	4, // 2: transform.TreeNode.nodes:type_name -> transform.TreeNode
+	6, // 0: transform.TreeNodes.nodes:type_name -> transform.TreeNode
+	7, // 1: transform.TreeNode.tag:type_name -> transform.Tag
+	6, // 2: transform.TreeNode.nodes:type_name -> transform.TreeNode
 	0, // 3: transform.Transform.GetUser:input_type -> transform.UserRequest
-	2, // 4: transform.Transform.GetErpCityTree:input_type -> transform.Empty
-	1, // 5: transform.Transform.GetUser:output_type -> transform.UserResponse
-	3, // 6: transform.Transform.GetErpCityTree:output_type -> transform.TreeNodes
-	5, // [5:7] is the sub-list for method output_type
-	3, // [3:5] is the sub-list for method input_type
+	2, // 4: transform.Transform.ParseToken:input_type -> transform.TokenRequest
+	4, // 5: transform.Transform.GetErpCityTree:input_type -> transform.Empty
+	1, // 6: transform.Transform.GetUser:output_type -> transform.UserResponse
+	3, // 7: transform.Transform.ParseToken:output_type -> transform.TokenResponse
+	5, // 8: transform.Transform.GetErpCityTree:output_type -> transform.TreeNodes
+	6, // [6:9] is the sub-list for method output_type
+	3, // [3:6] is the sub-list for method input_type
 	3, // [3:3] is the sub-list for extension type_name
 	3, // [3:3] is the sub-list for extension extendee
 	0, // [0:3] is the sub-list for field type_name
@@ -522,7 +629,7 @@ func file_transform_proto_init() {
 			}
 		}
 		file_transform_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} {
-			switch v := v.(*Empty); i {
+			switch v := v.(*TokenRequest); i {
 			case 0:
 				return &v.state
 			case 1:
@@ -534,7 +641,7 @@ func file_transform_proto_init() {
 			}
 		}
 		file_transform_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} {
-			switch v := v.(*TreeNodes); i {
+			switch v := v.(*TokenResponse); i {
 			case 0:
 				return &v.state
 			case 1:
@@ -546,7 +653,7 @@ func file_transform_proto_init() {
 			}
 		}
 		file_transform_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} {
-			switch v := v.(*TreeNode); i {
+			switch v := v.(*Empty); i {
 			case 0:
 				return &v.state
 			case 1:
@@ -558,6 +665,30 @@ func file_transform_proto_init() {
 			}
 		}
 		file_transform_proto_msgTypes[5].Exporter = func(v interface{}, i int) interface{} {
+			switch v := v.(*TreeNodes); i {
+			case 0:
+				return &v.state
+			case 1:
+				return &v.sizeCache
+			case 2:
+				return &v.unknownFields
+			default:
+				return nil
+			}
+		}
+		file_transform_proto_msgTypes[6].Exporter = func(v interface{}, i int) interface{} {
+			switch v := v.(*TreeNode); i {
+			case 0:
+				return &v.state
+			case 1:
+				return &v.sizeCache
+			case 2:
+				return &v.unknownFields
+			default:
+				return nil
+			}
+		}
+		file_transform_proto_msgTypes[7].Exporter = func(v interface{}, i int) interface{} {
 			switch v := v.(*Tag); i {
 			case 0:
 				return &v.state
@@ -576,7 +707,7 @@ func file_transform_proto_init() {
 			GoPackagePath: reflect.TypeOf(x{}).PkgPath(),
 			RawDescriptor: file_transform_proto_rawDesc,
 			NumEnums:      0,
-			NumMessages:   6,
+			NumMessages:   8,
 			NumExtensions: 0,
 			NumServices:   1,
 		},
@@ -603,6 +734,7 @@ const _ = grpc.SupportPackageIsVersion6
 // For semantics around ctx use and closing/ending streaming RPCs, please refer to https://godoc.org/google.golang.org/grpc#ClientConn.NewStream.
 type TransformClient interface {
 	GetUser(ctx context.Context, in *UserRequest, opts ...grpc.CallOption) (*UserResponse, error)
+	ParseToken(ctx context.Context, in *TokenRequest, opts ...grpc.CallOption) (*TokenResponse, error)
 	GetErpCityTree(ctx context.Context, in *Empty, opts ...grpc.CallOption) (*TreeNodes, error)
 }
 
@@ -623,6 +755,15 @@ func (c *transformClient) GetUser(ctx context.Context, in *UserRequest, opts ...
 	return out, nil
 }
 
+func (c *transformClient) ParseToken(ctx context.Context, in *TokenRequest, opts ...grpc.CallOption) (*TokenResponse, error) {
+	out := new(TokenResponse)
+	err := c.cc.Invoke(ctx, "/transform.Transform/ParseToken", in, out, opts...)
+	if err != nil {
+		return nil, err
+	}
+	return out, nil
+}
+
 func (c *transformClient) GetErpCityTree(ctx context.Context, in *Empty, opts ...grpc.CallOption) (*TreeNodes, error) {
 	out := new(TreeNodes)
 	err := c.cc.Invoke(ctx, "/transform.Transform/GetErpCityTree", in, out, opts...)
@@ -635,6 +776,7 @@ func (c *transformClient) GetErpCityTree(ctx context.Context, in *Empty, opts ..
 // TransformServer is the server API for Transform service.
 type TransformServer interface {
 	GetUser(context.Context, *UserRequest) (*UserResponse, error)
+	ParseToken(context.Context, *TokenRequest) (*TokenResponse, error)
 	GetErpCityTree(context.Context, *Empty) (*TreeNodes, error)
 }
 
@@ -645,6 +787,9 @@ type UnimplementedTransformServer struct {
 func (*UnimplementedTransformServer) GetUser(context.Context, *UserRequest) (*UserResponse, error) {
 	return nil, status.Errorf(codes.Unimplemented, "method GetUser not implemented")
 }
+func (*UnimplementedTransformServer) ParseToken(context.Context, *TokenRequest) (*TokenResponse, error) {
+	return nil, status.Errorf(codes.Unimplemented, "method ParseToken not implemented")
+}
 func (*UnimplementedTransformServer) GetErpCityTree(context.Context, *Empty) (*TreeNodes, error) {
 	return nil, status.Errorf(codes.Unimplemented, "method GetErpCityTree not implemented")
 }
@@ -671,6 +816,24 @@ func _Transform_GetUser_Handler(srv interface{}, ctx context.Context, dec func(i
 	return interceptor(ctx, in, info, handler)
 }
 
+func _Transform_ParseToken_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
+	in := new(TokenRequest)
+	if err := dec(in); err != nil {
+		return nil, err
+	}
+	if interceptor == nil {
+		return srv.(TransformServer).ParseToken(ctx, in)
+	}
+	info := &grpc.UnaryServerInfo{
+		Server:     srv,
+		FullMethod: "/transform.Transform/ParseToken",
+	}
+	handler := func(ctx context.Context, req interface{}) (interface{}, error) {
+		return srv.(TransformServer).ParseToken(ctx, req.(*TokenRequest))
+	}
+	return interceptor(ctx, in, info, handler)
+}
+
 func _Transform_GetErpCityTree_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
 	in := new(Empty)
 	if err := dec(in); err != nil {
@@ -697,6 +860,10 @@ var _Transform_serviceDesc = grpc.ServiceDesc{
 			MethodName: "GetUser",
 			Handler:    _Transform_GetUser_Handler,
 		},
+		{
+			MethodName: "ParseToken",
+			Handler:    _Transform_ParseToken_Handler,
+		},
 		{
 			MethodName: "GetErpCityTree",
 			Handler:    _Transform_GetErpCityTree_Handler,

+ 17 - 5
transformclient/transform.go

@@ -10,16 +10,23 @@ import (
 
 	"git.i2edu.net/i2/i2-bill-erp/transform"
 
-	"git.i2edu.net/i2/go-zero/zrpc"
+	"github.com/tal-tech/go-zero/zrpc"
 )
 
 type (
-	UserRequest  = transform.UserRequest
-	UserResponse = transform.UserResponse
+	TreeNodes     = transform.TreeNodes
+	TreeNode      = transform.TreeNode
+	Tag           = transform.Tag
+	UserRequest   = transform.UserRequest
+	UserResponse  = transform.UserResponse
+	TokenRequest  = transform.TokenRequest
+	TokenResponse = transform.TokenResponse
+	Empty         = transform.Empty
 
 	Transform interface {
 		GetUser(ctx context.Context, in *UserRequest) (*UserResponse, error)
-		GetErpCityTree(ctx context.Context, in *transform.Empty) (*transform.TreeNodes, error)
+		ParseToken(ctx context.Context, in *TokenRequest) (*TokenResponse, error)
+		GetErpCityTree(ctx context.Context, in *Empty) (*TreeNodes, error)
 	}
 
 	defaultTransform struct {
@@ -38,7 +45,12 @@ func (m *defaultTransform) GetUser(ctx context.Context, in *UserRequest) (*UserR
 	return client.GetUser(ctx, in)
 }
 
-func (m *defaultTransform) GetErpCityTree(ctx context.Context, in *transform.Empty) (*transform.TreeNodes, error) {
+func (m *defaultTransform) ParseToken(ctx context.Context, in *TokenRequest) (*TokenResponse, error) {
+	client := transform.NewTransformClient(m.cli.Conn())
+	return client.ParseToken(ctx, in)
+}
+
+func (m *defaultTransform) GetErpCityTree(ctx context.Context, in *Empty) (*TreeNodes, error) {
 	client := transform.NewTransformClient(m.cli.Conn())
 	return client.GetErpCityTree(ctx, in)
 }