123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270 |
- package bus
- import (
- "encoding/hex"
- "fmt"
- "net"
- "reflect"
- "strconv"
- "sync"
- "time"
- "github.com/yuguorong/go/log"
- "github.com/ammeter/config"
- )
- const (
- maxConnCount = 100
- moduleName = "dtuFtpServer"
- DEF_FTP_PORT = 10010
- )
- var cmdQueryAddr []byte = []byte{0xfe, 0xfe, 0xfe, 0xfe, 0x68, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0xAA, 0x68, 0x13, 0x00, 0xDF, 0x16}
- type DtuServer struct {
- baseBus
- Port int
- name string
- lis net.Listener
- lisAlt net.Listener
- connlist map[int]net.Conn
- clientNum int
- loop bool
- }
- func (dtu *DtuServer) ResetChannel(chn IChannel) {
- }
- func (dtu *DtuServer) closeConn(conn net.Conn, connid int, chn IChannel) {
- dtu.mutex.Lock()
- if _, has := dtu.connlist[connid]; has {
- delete(dtu.connlist, connid)
- dtu.clientNum--
- }
- dtu.mutex.Unlock()
- conn.Close()
- if chn != nil {
- basechn := chn.(*BusChannel)
- log.Infof("client [%s] close\n", basechn.chnID)
- basechn.mountCnt = 0
- if basechn.event != nil {
- basechn.event.OnDetach(chn)
- }
- for i, id := range basechn.conIdList {
- if id == connid {
- basechn.conIdList[i] = -1
- break
- }
- }
- }
- }
- func (dtu *DtuServer) LookupChannel(buf []byte, idx int, conn net.Conn) IChannel {
- return dtu.baseBus.ScanChannel(buf, idx)
- }
- func (dtu *DtuServer) ClientConnect(conn net.Conn, idx int, altFlag bool) {
- var ic IChannel = nil
- remoteAddr := conn.RemoteAddr().String()
- //SrcPort := uint(conn.RemoteAddr().(*net.TCPAddr).Port)
- defer dtu.closeConn(conn, idx, ic)
- var err error
- var buf [1024]byte
- n := 0
- for ic == nil {
- if altFlag {
- conn.Write(cmdQueryAddr)
- }
- n, err = conn.Read(buf[:])
- if err != nil {
- log.Infof("read from %s header faild err:[%v]\n", remoteAddr, err)
- return
- }
- log.Infof("Connected(alternat port:%t) with code %X", altFlag, buf[:n])
- ic = dtu.LookupChannel(buf[:n], idx, conn)
- }
- chnin := ic.GetChan(0)
- if chnin == nil {
- panic("no chan for read message")
- }
- chnout := ic.GetChan(1)
- dtu.name = hex.EncodeToString(buf[:n])
- for {
- smeter := hex.EncodeToString(buf[:n])
- dtu.mutex.Lock()
- chnin <- buf[:n]
- dtu.mutex.Unlock()
- if chnout != nil {
- after := time.After(time.Second * 60)
- select {
- case msg := <-chnout:
- conn.Write(msg.([]byte))
- case <-after:
- break
- }
- }
- // log.Printf("rev data from %s msg:%s\n", conn.RemoteAddr().String(), string(buf[:n]))
- log.Infof("[%s]rev data from %s msg(%d):[%s]\n", time.Now().Format("2006-01-02 15:04:05"), remoteAddr, n, smeter)
- if ic.(*BusChannel).timeout != 0 {
- conn.SetReadDeadline(time.Now().Add(ic.(*BusChannel).timeout))
- }
- n, err = conn.Read(buf[:])
- if err != nil {
- log.Errorf("read from %s msg faild err:[%v]\n", ic.ID(), err)
- dtu.closeConn(conn, idx, ic)
- break
- }
- }
- }
- func (dtu *DtuServer) StartServer(lister *net.Listener, altFlag bool) {
- defer func() {
- (*lister).Close()
- }()
- idx := 0
- for dtu.loop {
- if dtu.clientNum >= maxConnCount {
- log.Infof("there is %d clients,is max num\n", dtu.clientNum)
- time.Sleep(5 * time.Second)
- continue
- }
- conn, err := (*lister).Accept()
- if err != nil {
- log.Errorf("listen err:[%v]\n", err)
- }
- dtu.mutex.Lock()
- dtu.connlist[idx] = conn
- idx++
- dtu.clientNum++
- dtu.mutex.Unlock()
- go dtu.ClientConnect(conn, idx-1, altFlag)
- }
- }
- func (dtu *DtuServer) Init() error {
- dtu.baseBus.Init()
- if !dtu.loop {
- addr := fmt.Sprintf("0.0.0.0:%d", dtu.Port)
- log.Info("start ", addr)
- var err error = nil
- dtu.lis, err = net.Listen("tcp", addr)
- if err != nil {
- log.Errorf("err! addr %s open faild err:[%v]\n", addr, err)
- return err
- }
- dtu.mutex.Lock()
- defer dtu.mutex.Unlock()
- dtu.loop = true
- go dtu.StartServer(&dtu.lis, false)
- addr = fmt.Sprintf("0.0.0.0:%d", dtu.Port+10)
- log.Info("Alternat port listen start ", addr)
- dtu.lisAlt, err = net.Listen("tcp", addr)
- if err != nil {
- log.Errorf("err! addr %s open faild err:[%v]\n", addr, err)
- return err
- }
- go dtu.StartServer(&dtu.lisAlt, true)
- }
- return nil
- }
- func (dtu *DtuServer) Uninit() {
- dtu.mutex.Lock()
- defer dtu.mutex.Unlock()
- if dtu.loop {
- dtu.loop = false
- for _, c := range dtu.connlist {
- c.Close()
- }
- dtu.lis.Close()
- dtu.connlist = make(map[int]net.Conn)
- }
- }
- func (dtu *DtuServer) OpenChannel(chn interface{}, router []chan interface{}) IChannel {
- if chn == nil || reflect.TypeOf(chn).Kind() != reflect.String {
- panic("Open channel should be a uniq string")
- }
- ic := dtu.baseBus.OpenChannel(chn, router)
- return ic
- }
- func (dtu *DtuServer) CloseChannel(chn IChannel) error {
- dtu.baseBus.CloseChannel(chn)
- if dtu.loop {
- //idx := chn.(*BusChannel).conn.(int)
- for _, connID := range chn.(*BusChannel).conIdList {
- if conn, has := dtu.connlist[connID]; has {
- dtu.mutex.Lock()
- delete(dtu.connlist, connID)
- dtu.mutex.Unlock()
- conn.Close()
- }
- }
- }
- return nil
- }
- func (dtu *DtuServer) Send(ichn IChannel, buff interface{}) (int, error) {
- if ichn != nil {
- chn := ichn.(*BusChannel)
- for _, connID := range chn.conIdList {
- if connID >= 0 {
- if conn, has := dtu.connlist[connID]; has {
- conn.Write(buff.([]byte))
- }
- }
- }
- }
- return dtu.baseBus.Send(ichn, buff)
- }
- func GetFtpServerConfig(param []interface{}) int {
- Port := 0
- if param != nil && len(param) > 0 && reflect.TypeOf(param[0]).Kind() == reflect.Int {
- Port = param[0].(int)
- }
- if Port == 0 {
- Port = config.GetSysConfig().GetValue("Bus/DtuServer/Port", DEF_FTP_PORT).(int)
- }
- return Port
- }
- func NewDtuServer(param []interface{}) IBus {
- Port := GetFtpServerConfig(param)
- busid := GenDtuServerId(moduleName, param)
- b := &DtuServer{
- baseBus: baseBus{
- BusId: busid,
- mutex: &sync.Mutex{},
- },
- connlist: make(map[int]net.Conn),
- clientNum: 0,
- Port: Port,
- loop: false,
- }
- return b
- }
- func GenDtuServerId(name string, param []interface{}) string {
- Port := GetFtpServerConfig(param)
- return name + ":" + strconv.Itoa(Port)
- }
- func init() {
- BusReg[moduleName] = NewDtuServer
- BusGetID[moduleName] = GenDtuServerId
- }
|