本文整理匯總了Golang中github.com/wfxiang08/rpc_proxy/utils/log.Printf函數的典型用法代碼示例。如果您正苦於以下問題:Golang Printf函數的具體用法?Golang Printf怎麽用?Golang Printf使用的例子?那麽, 這裏精選的函數代碼示例或許可以為您提供幫助。
在下文中一共展示了Printf函數的15個代碼示例,這些例子默認根據受歡迎程度排序。您可以為喜歡或者感覺有用的代碼點讚,您的評價將有助於係統推薦出更棒的Golang代碼示例。
示例1: nextBackendConn
// 獲取下一個active狀態的BackendConn
func (s *BackServiceLB) nextBackendConn() *BackendConnLB {
s.activeConnsLock.Lock()
defer s.activeConnsLock.Unlock()
// TODO: 暫時采用RoundRobin的方法,可以采用其他具有優先級排列的方法
var backSocket *BackendConnLB
if len(s.activeConns) == 0 {
if s.verbose {
log.Printf(Cyan("[%s]ActiveConns Len 0"), s.serviceName)
}
backSocket = nil
} else {
if s.currentConnIndex >= len(s.activeConns) {
s.currentConnIndex = 0
}
backSocket = s.activeConns[s.currentConnIndex]
s.currentConnIndex++
if s.verbose {
log.Printf(Cyan("[%s]ActiveConns Len %d, CurrentIndex: %d"), s.serviceName,
len(s.activeConns), s.currentConnIndex)
}
}
return backSocket
}
示例2: flushRequests
// 處理所有的等待中的請求
func (bc *BackendConn) flushRequests(err error) {
// 告訴BackendService, 不再接受新的請求
bc.MarkConnActiveFalse()
bc.Lock()
seqRequest := bc.seqNum2Request
bc.seqNum2Request = make(map[int32]*Request, 4096)
bc.Unlock()
threshold := time.Now().Add(-time.Second * 5)
for _, request := range seqRequest {
if request.Start > 0 {
t := time.Unix(request.Start, 0)
if t.After(threshold) {
// 似乎在筆記本上,合上顯示器之後出出現網絡錯誤
log.Printf(Red("[%s]Handle Failed Request: %s, Started: %s"),
request.Service, request.Request.Name, FormatYYYYmmDDHHMMSS(t))
}
} else {
log.Printf(Red("[%s]Handle Failed Request: %s"), request.Service,
request.Request.Name)
}
request.Response.Err = err
if request.Wait != nil {
request.Wait.Done()
}
}
}
示例3: setResponse
// 配對 Request, resp, err
// PARAM: resp []byte 為一幀完整的thrift數據包
func (bc *BackendConn) setResponse(r *Request, data []byte, err error) error {
// 表示出現錯誤了
if data == nil {
log.Printf("[%s]No Data From Server, error: %v", r.Service, err)
r.Response.Err = err
} else {
// 從resp中讀取基本的信息
typeId, seqId, err := DecodeThriftTypIdSeqId(data)
// 解碼錯誤,直接報錯
if err != nil {
return err
}
// 找到對應的Request
bc.Lock()
req, ok := bc.seqNum2Request[seqId]
if ok {
delete(bc.seqNum2Request, seqId)
}
bc.Unlock()
// 如果是心跳,則OK
if typeId == MESSAGE_TYPE_HEART_BEAT {
// log.Printf(Magenta("Get Ping/Pang Back"))
bc.hbLastTime.Set(time.Now().Unix())
return nil
}
if !ok {
return errors.New("Invalid Response")
}
if bc.verbose {
log.Printf("[%s]Data From Server, seqId: %d, Request: %d", req.Service, seqId, req.Request.SeqId)
}
r = req
r.Response.TypeId = typeId
}
r.Response.Data, r.Response.Err = data, err
// 還原SeqId
if data != nil {
r.RestoreSeqId()
}
// 設置幾個控製用的channel
if err != nil && r.Failed != nil {
r.Failed.Set(true)
}
if r.Wait != nil {
r.Wait.Done()
}
return err
}
示例4: Dispatch
func (p *fakeServer) Dispatch(r *Request) error {
log.Printf("Request SeqId: %d, MethodName: %s\n", r.Request.SeqId, r.Request.Name)
r.Wait.Add(1)
go func() {
time.Sleep(time.Millisecond)
r.Response.Data = []byte(string(r.Request.Data))
typeId, seqId, _ := DecodeThriftTypIdSeqId(r.Response.Data)
log.Printf(Green("TypeId: %d, SeqId: %d\n"), typeId, seqId)
r.Wait.Done()
}()
// r.RestoreSeqId()
// r.Wait.Done()
return nil
}
示例5: Run
//
// 不斷建立到後端的邏輯,負責: BackendConn#input到redis的數據的輸入和返回
//
func (bc *BackendConn) Run() {
for k := 0; !bc.IsMarkOffline.Get(); k++ {
// 1. 首先BackendConn將當前 input中的數據寫到後端服務中
transport, err := bc.ensureConn()
if err != nil {
log.ErrorErrorf(err, "[%s]BackendConn#ensureConn error: %v", bc.service, err)
return
}
c := NewTBufferedFramedTransport(transport, 100*time.Microsecond, 20)
// 2. 將 bc.input 中的請求寫入 後端的Rpc Server
err = bc.loopWriter(c) // 同步
// 3. 停止接受Request
bc.MarkConnActiveFalse()
// 4. 將bc.input中剩餘的 Request直接出錯處理
if err == nil {
log.Printf(Red("[%s]BackendConn#loopWriter normal Exit..."), bc.service)
break
} else {
// 對於尚未處理的Request, 直接報錯
for i := len(bc.input); i != 0; i-- {
r := <-bc.input
bc.setResponse(r, nil, err)
}
}
}
}
示例6: flushRequests
// 處理所有的等待中的請求
func (bc *BackendConnLB) flushRequests(err error) {
// 告訴BackendService, 不再接受新的請求
bc.MarkConnActiveFalse()
bc.Lock()
seqRequest := bc.seqNum2Request
bc.seqNum2Request = make(map[int32]*Request)
bc.Unlock()
for _, request := range seqRequest {
if request.Request.TypeId == MESSAGE_TYPE_HEART_BEAT {
// 心跳出錯了,則直接直接跳過
} else {
log.Printf(Red("Handle Failed Request: %s.%s"), request.Service, request.Request.Name)
request.Response.Err = err
if request.Wait != nil {
request.Wait.Done()
}
}
}
// 關閉輸入
close(bc.input)
}
示例7: Dispatch
//
// 後端如何處理一個Request, 處理完畢之後直接返回,因為Caller已經做好異步處理了
//
func (s *BackServiceLB) Dispatch(r *Request) error {
backendConn := s.nextBackendConn()
r.Service = s.serviceName
if backendConn == nil {
// 沒有後端服務
if s.verbose {
log.Printf(Red("[%s]No BackSocket Found: %s"),
s.serviceName, r.Request.Name)
}
// 從errMsg來構建異常
errMsg := GetWorkerNotFoundData(r, "BackServiceLB")
// log.Printf(Magenta("---->Convert Error Back to Exception:[%d] %s\n"), len(errMsg), string(errMsg))
r.Response.Data = errMsg
return nil
} else {
// if s.verbose {
// log.Println("SendMessage With: ", backendConn.Addr4Log(), "For Service: ", s.serviceName)
// }
backendConn.PushBack(r)
r.Wait.Wait()
return nil
}
}
示例8: loopWriter
func (s *Session) loopWriter(tasks <-chan *Request) error {
// Proxy: Session ---> Client
for r := range tasks {
// 1. 等待Request對應的Response
// 出錯了如何處理呢?
s.handleResponse(r)
// 2. 將結果寫回給Client
if s.verbose {
log.Printf("[%s]Session#loopWriter --> client FrameSize: %d",
r.Service, len(r.Response.Data))
}
// r.Response.Data ---> Client
_, err := s.TBufferedFramedTransport.Write(r.Response.Data)
if err != nil {
log.ErrorErrorf(err, "Write back Data Error: %v", err)
return err
}
// 3. Flush
err = s.TBufferedFramedTransport.FlushBuffer(true) // len(tasks) == 0
if err != nil {
log.ErrorErrorf(err, "Write back Data Error: %v", err)
return err
}
r.Recycle()
}
return nil
}
示例9: setResponse
// 配對 Request, resp, err
// PARAM: resp []byte 為一幀完整的thrift數據包
func (bc *BackendConnLB) setResponse(r *Request, data []byte, err error) error {
// 表示出現錯誤了
if data == nil {
log.Printf("No Data From Server, error: %v\n", err)
r.Response.Err = err
} else {
// 從resp中讀取基本的信息
typeId, seqId, err := DecodeThriftTypIdSeqId(data)
// 解碼錯誤,直接報錯
if err != nil {
return err
}
if typeId == MESSAGE_TYPE_STOP {
// 不再接受新的輸入
// 直接來自後端的服務(不遵循: Request/Reply模型)
bc.MarkConnActiveFalse()
return nil
}
// 找到對應的Request
bc.Lock()
req, ok := bc.seqNum2Request[seqId]
if ok {
delete(bc.seqNum2Request, seqId)
}
bc.Unlock()
// 如果是心跳,則OK
if typeId == MESSAGE_TYPE_HEART_BEAT {
bc.hbLastTime.Set(time.Now().Unix())
return nil
}
if !ok {
return errors.New("Invalid Response")
}
// log.Printf("Data From Server, seqId: %d, Request: %d\n", seqId, req.Request.SeqId)
r = req
r.Response.TypeId = typeId
}
r.Response.Data, r.Response.Err = data, err
// 還原SeqId
if data != nil {
r.RestoreSeqId()
}
// 設置幾個控製用的channel
if err != nil && r.Failed != nil {
r.Failed.Set(true)
}
if r.Wait != nil {
r.Wait.Done()
}
return err
}
示例10: ensureConn
//
// 確保Socket成功連接到後端服務器
//
func (bc *BackendConn) ensureConn() (transport thrift.TTransport, err error) {
// 1. 創建連接(隻要IP沒有問題, err一般就是空)
timeout := time.Second * 5
if strings.Contains(bc.addr, ":") {
transport, err = thrift.NewTSocketTimeout(bc.addr, timeout)
} else {
transport, err = NewTUnixDomainTimeout(bc.addr, timeout)
}
log.Printf(Cyan("[%s]Create Socket To: %s"), bc.service, bc.addr)
if err != nil {
log.ErrorErrorf(err, "[%s]Create Socket Failed: %v, Addr: %s", err, bc.service, bc.addr)
// 連接不上,失敗
return nil, err
}
// 2. 隻要服務存在,一般不會出現err
sleepInterval := 1
err = transport.Open()
for err != nil && !bc.IsMarkOffline.Get() {
log.ErrorErrorf(err, "[%s]Socket Open Failed: %v, Addr: %s", bc.service, err, bc.addr)
// Sleep: 1, 2, 4這幾個間隔
time.Sleep(time.Duration(sleepInterval) * time.Second)
if sleepInterval < 4 {
sleepInterval *= 2
}
err = transport.Open()
}
return transport, err
}
示例11: Run
//
// 兩參數是必須的: ProductName, zkAddress, frontAddr可以用來測試
//
func (p *ProxyServer) Run() {
var transport thrift.TServerTransport
var err error
log.Printf(Magenta("Start Proxy at Address: %s"), p.proxyAddr)
// 讀取後端服務的配置
isUnixDomain := false
if !strings.Contains(p.proxyAddr, ":") {
if FileExist(p.proxyAddr) {
os.Remove(p.proxyAddr)
}
transport, err = NewTServerUnixDomain(p.proxyAddr)
isUnixDomain = true
} else {
transport, err = thrift.NewTServerSocket(p.proxyAddr)
}
if err != nil {
log.ErrorErrorf(err, "Server Socket Create Failed: %v, Front: %s", err, p.proxyAddr)
}
// 開始監聽
// transport.Open()
transport.Listen()
ch := make(chan thrift.TTransport, 4096)
defer close(ch)
go func() {
var address string
for c := range ch {
// 為每個Connection建立一個Session
socket, ok := c.(SocketAddr)
if isUnixDomain {
address = p.proxyAddr
} else if ok {
address = socket.Addr().String()
} else {
address = "unknow"
}
x := NewSession(c, address, p.verbose)
// Session獨立處理自己的請求
go x.Serve(p.router, 1000)
}
}()
// Accept什麽時候出錯,出錯之後如何處理呢?
for {
c, err := transport.Accept()
if err != nil {
log.ErrorErrorf(err, "Accept Error: %v", err)
break
} else {
ch <- c
}
}
}
示例12: StateChanged
func (s *BackService) StateChanged(conn *BackendConn) {
log.Printf(Cyan("[%s]StateChanged: %s, Index: %d, Count: %d, IsConnActive: %t"),
s.serviceName, conn.addr, conn.Index, len(s.activeConns),
conn.IsConnActive.Get())
s.activeConnsLock.Lock()
defer s.activeConnsLock.Unlock()
if conn.IsConnActive.Get() {
log.Printf(Cyan("[%s]MarkConnActiveOK: %s, Index: %d, Count: %d"),
s.serviceName, conn.addr, conn.Index, len(s.activeConns))
if conn.Index == INVALID_ARRAY_INDEX {
conn.Index = len(s.activeConns)
s.activeConns = append(s.activeConns, conn)
log.Printf(Green("[%s]Add BackendConn to activeConns: %s, Total Actives: %d"),
s.serviceName, conn.Addr(), len(s.activeConns))
}
} else {
log.Printf(Red("[%s]Remove BackendConn From activeConns: %s, Index: %d"),
s.serviceName, conn.Addr(), conn.Index)
if conn.Index != INVALID_ARRAY_INDEX {
lastIndex := len(s.activeConns) - 1
// 將最後一個元素和當前的元素交換位置
if lastIndex != conn.Index {
lastConn := s.activeConns[lastIndex]
s.activeConns[conn.Index] = lastConn
lastConn.Index = conn.Index
}
s.activeConns[lastIndex] = nil
conn.Index = INVALID_ARRAY_INDEX
// slice
s.activeConns = s.activeConns[0:lastIndex]
log.Printf(Red("[%s]Remove BackendConn From activeConns: %s, Remains: %d"),
s.serviceName, conn.Addr(), len(s.activeConns))
}
}
}
示例13: MarkOffline
//
// MarkOffline發生場景:
// 1. 後端服務即將下線,預先通知
// 2. 後端服務已經掛了,zk檢測到
//
// BackendConn 在這裏暫時理解關閉conn, 而是從 backend_service_proxy中下線當前的conn,
// 然後conn的關閉根據 心跳&Conn的讀寫異常來判斷; 因此 IsConnActive = false 情況下,心跳不能關閉
//
func (bc *BackendConn) MarkOffline() {
if !bc.IsMarkOffline.Get() {
log.Printf(Magenta("[%s]BackendConn: %s MarkOffline"), bc.service, bc.addr)
bc.IsMarkOffline.Set(true)
// 不再接受(來自backend_service_proxy的)新的輸入
bc.MarkConnActiveFalse()
close(bc.input)
}
}
示例14: MarkConnActiveFalse
func (bc *BackendConn) MarkConnActiveFalse() {
if bc.IsConnActive.Get() {
log.Printf(Red("[%s]MarkConnActiveFalse: %s, %p"), bc.service, bc.addr, bc.delegate)
// 從Active切換到非正常狀態
bc.IsConnActive.Set(false)
if bc.delegate != nil {
bc.delegate.StateChanged(bc) // 通知其他人狀態出現問題
}
}
}
示例15: StateChanged
// 隻有在conn出現錯誤時才會調用
func (s *BackServiceLB) StateChanged(conn *BackendConnLB) {
s.activeConnsLock.Lock()
defer s.activeConnsLock.Unlock()
log.Printf(Green("[%s]StateChanged: %s, Index: %d, Count: %d"), conn.serviceName, conn.addr4Log, conn.Index, len(s.activeConns))
if conn.IsConnActive.Get() {
// BackServiceLB 隻有一個狀態轉移: Active --> Not Active
log.Printf(Magenta("Unexpected BackendConnLB State"))
if s.verbose {
panic("Unexpected BackendConnLB State")
}
} else {
log.Printf(Red("Remove BackendConn From activeConns: %s, Index: %d, Count: %d"),
conn.Addr4Log(), conn.Index, len(s.activeConns))
// 從數組中刪除一個元素(O(1)的操作)
if conn.Index != INVALID_ARRAY_INDEX {
// 1. 和最後一個元素進行交換
lastIndex := len(s.activeConns) - 1
if lastIndex != conn.Index {
lastConn := s.activeConns[lastIndex]
// 將最後一個元素和當前的元素交換位置
s.activeConns[conn.Index] = lastConn
lastConn.Index = conn.Index
// 刪除引用
s.activeConns[lastIndex] = nil
conn.Index = INVALID_ARRAY_INDEX
}
log.Printf(Red("Remove BackendConn From activeConns: %s"), conn.Addr4Log())
// 2. slice
s.activeConns = s.activeConns[0:lastIndex]
}
}
}