Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 1 | /* |
| 2 | ================================================================================== |
| 3 | Copyright (c) 2019 AT&T Intellectual Property. |
| 4 | Copyright (c) 2019 Nokia |
| 5 | |
| 6 | Licensed under the Apache License, Version 2.0 (the "License"); |
| 7 | you may not use this file except in compliance with the License. |
| 8 | You may obtain a copy of the License at |
| 9 | |
| 10 | http://www.apache.org/licenses/LICENSE-2.0 |
| 11 | |
| 12 | Unless required by applicable law or agreed to in writing, software |
| 13 | distributed under the License is distributed on an "AS IS" BASIS, |
| 14 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 15 | See the License for the specific language governing permissions and |
| 16 | limitations under the License. |
| 17 | ================================================================================== |
| 18 | */ |
| 19 | |
| 20 | package xapp |
| 21 | |
| 22 | /* |
| 23 | #include <time.h> |
| 24 | #include <stdlib.h> |
| 25 | #include <stdio.h> |
| 26 | #include <string.h> |
| 27 | #include <rmr/rmr.h> |
| 28 | #include <rmr/RIC_message_types.h> |
| 29 | |
| 30 | void write_bytes_array(unsigned char *dst, void *data, int len) { |
| 31 | memcpy((void *)dst, (void *)data, len); |
| 32 | } |
| 33 | |
| 34 | #cgo CFLAGS: -I../ |
| 35 | #cgo LDFLAGS: -lrmr_nng -lnng |
| 36 | */ |
| 37 | import "C" |
| 38 | |
| 39 | import ( |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 40 | "fmt" |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 41 | "github.com/spf13/viper" |
| 42 | "strconv" |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 43 | "strings" |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 44 | "time" |
| 45 | "unsafe" |
| 46 | ) |
| 47 | |
| 48 | var RMRCounterOpts = []CounterOpts{ |
| 49 | {Name: "Transmitted", Help: "The total number of transmited RMR messages"}, |
| 50 | {Name: "Received", Help: "The total number of received RMR messages"}, |
| 51 | {Name: "TransmitError", Help: "The total number of RMR transmission errors"}, |
| 52 | {Name: "ReceiveError", Help: "The total number of RMR receive errors"}, |
| 53 | } |
| 54 | |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 55 | var RMRErrors = map[int]string{ |
| 56 | C.RMR_OK: "state is good", |
| 57 | C.RMR_ERR_BADARG: "argument passed to function was unusable", |
| 58 | C.RMR_ERR_NOENDPT: "send/call could not find an endpoint based on msg type", |
| 59 | C.RMR_ERR_EMPTY: "msg received had no payload; attempt to send an empty message", |
| 60 | C.RMR_ERR_NOHDR: "message didn't contain a valid header", |
| 61 | C.RMR_ERR_SENDFAILED: "send failed; errno has nano reason", |
| 62 | C.RMR_ERR_CALLFAILED: "unable to send call() message", |
| 63 | C.RMR_ERR_NOWHOPEN: "no wormholes are open", |
| 64 | C.RMR_ERR_WHID: "wormhole id was invalid", |
| 65 | C.RMR_ERR_OVERFLOW: "operation would have busted through a buffer/field size", |
| 66 | C.RMR_ERR_RETRY: "request (send/call/rts) failed, but caller should retry (EAGAIN for wrappers)", |
| 67 | C.RMR_ERR_RCVFAILED: "receive failed (hard error)", |
| 68 | C.RMR_ERR_TIMEOUT: "message processing call timed out", |
| 69 | C.RMR_ERR_UNSET: "the message hasn't been populated with a transport buffer", |
| 70 | C.RMR_ERR_TRUNC: "received message likely truncated", |
| 71 | C.RMR_ERR_INITFAILED: "initialization of something (probably message) failed", |
| 72 | C.RMR_ERR_NOTSUPP: "the request is not supported, or RMr was not initialized for the request", |
| 73 | } |
| 74 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 75 | type RMRParams struct { |
Mohamed Abukar | 19461e1 | 2019-08-23 08:46:11 +0300 | [diff] [blame] | 76 | Mtype int |
| 77 | Payload []byte |
| 78 | PayloadLen int |
| 79 | Meid *RMRMeid |
| 80 | Xid string |
| 81 | SubId int |
| 82 | Src string |
| 83 | Mbuf *C.rmr_mbuf_t |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 84 | } |
| 85 | |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 86 | func NewRMRClient() *RMRClient { |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 87 | p := C.CString(viper.GetString("rmr.protPort")) |
| 88 | m := C.int(viper.GetInt("rmr.maxSize")) |
| 89 | defer C.free(unsafe.Pointer(p)) |
| 90 | |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 91 | ctx := C.rmr_init(p, m, C.int(0)) |
| 92 | if ctx == nil { |
Mohamed Abukar | 775722c | 2019-06-10 16:41:57 +0300 | [diff] [blame] | 93 | Logger.Error("rmrClient: Initializing RMR context failed, bailing out!") |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 94 | } |
| 95 | |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 96 | return &RMRClient{ |
Mohamed Abukar | 775722c | 2019-06-10 16:41:57 +0300 | [diff] [blame] | 97 | context: ctx, |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 98 | consumers: make([]MessageConsumer, 0), |
Mohamed Abukar | 775722c | 2019-06-10 16:41:57 +0300 | [diff] [blame] | 99 | stat: Metric.RegisterCounterGroup(RMRCounterOpts, "RMR"), |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 100 | } |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 101 | } |
| 102 | |
| 103 | func (m *RMRClient) Start(c MessageConsumer) { |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 104 | if c != nil { |
| 105 | m.consumers = append(m.consumers, c) |
| 106 | } |
| 107 | |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 108 | for { |
| 109 | Logger.Info("rmrClient: Waiting for RMR to be ready ...") |
| 110 | |
| 111 | if m.ready = int(C.rmr_ready(m.context)); m.ready == 1 { |
| 112 | break |
| 113 | } |
| 114 | time.Sleep(10 * time.Second) |
| 115 | } |
| 116 | m.wg.Add(viper.GetInt("rmr.numWorkers")) |
| 117 | |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 118 | if m.readyCb != nil { |
| 119 | go m.readyCb(m.readyCbParams) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 120 | } |
| 121 | |
| 122 | for w := 0; w < viper.GetInt("rmr.numWorkers"); w++ { |
| 123 | go m.Worker("worker-"+strconv.Itoa(w), 0) |
| 124 | } |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 125 | m.Wait() |
| 126 | } |
| 127 | |
| 128 | func (m *RMRClient) Worker(taskName string, msgSize int) { |
| 129 | p := viper.GetString("rmr.protPort") |
| 130 | Logger.Info("rmrClient: '%s': receiving messages on [%s]", taskName, p) |
| 131 | |
| 132 | defer m.wg.Done() |
| 133 | for { |
| 134 | rxBuffer := C.rmr_rcv_msg(m.context, nil) |
| 135 | if rxBuffer == nil { |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 136 | m.LogMBufError("RecvMsg failed", rxBuffer) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 137 | m.UpdateStatCounter("ReceiveError") |
| 138 | continue |
| 139 | } |
| 140 | m.UpdateStatCounter("Received") |
| 141 | |
| 142 | go m.parseMessage(rxBuffer) |
| 143 | } |
| 144 | } |
| 145 | |
| 146 | func (m *RMRClient) parseMessage(rxBuffer *C.rmr_mbuf_t) { |
| 147 | if len(m.consumers) == 0 { |
| 148 | Logger.Info("rmrClient: No message handlers defined, message discarded!") |
| 149 | return |
| 150 | } |
| 151 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 152 | params := &RMRParams{} |
| 153 | params.Mbuf = rxBuffer |
| 154 | params.Mtype = int(rxBuffer.mtype) |
| 155 | params.SubId = int(rxBuffer.sub_id) |
| 156 | params.Meid = &RMRMeid{} |
| 157 | |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 158 | meidBuf := make([]byte, int(C.RMR_MAX_MEID)) |
| 159 | if meidCstr := C.rmr_get_meid(rxBuffer, (*C.uchar)(unsafe.Pointer(&meidBuf[0]))); meidCstr != nil { |
Mohamed Abukar | 19461e1 | 2019-08-23 08:46:11 +0300 | [diff] [blame] | 160 | params.Meid.RanName = strings.TrimRight(string(meidBuf), "\000") |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 161 | } |
| 162 | |
| 163 | xidBuf := make([]byte, int(C.RMR_MAX_XID)) |
| 164 | if xidCstr := C.rmr_get_xact(rxBuffer, (*C.uchar)(unsafe.Pointer(&xidBuf[0]))); xidCstr != nil { |
| 165 | params.Xid = strings.TrimRight(string(xidBuf[0:32]), "\000") |
| 166 | } |
| 167 | |
| 168 | srcBuf := make([]byte, int(C.RMR_MAX_SRC)) |
| 169 | if srcStr := C.rmr_get_src(rxBuffer, (*C.uchar)(unsafe.Pointer(&srcBuf[0]))); srcStr != nil { |
| 170 | params.Src = strings.TrimRight(string(srcBuf[0:64]), "\000") |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 171 | } |
| 172 | |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 173 | for _, c := range m.consumers { |
| 174 | cptr := unsafe.Pointer(rxBuffer.payload) |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 175 | params.Payload = C.GoBytes(cptr, C.int(rxBuffer.len)) |
| 176 | params.PayloadLen = int(rxBuffer.len) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 177 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 178 | err := c.Consume(params) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 179 | if err != nil { |
| 180 | Logger.Warn("rmrClient: Consumer returned error: %v", err) |
| 181 | } |
| 182 | } |
| 183 | } |
| 184 | |
| 185 | func (m *RMRClient) Allocate() *C.rmr_mbuf_t { |
| 186 | buf := C.rmr_alloc_msg(m.context, 0) |
| 187 | if buf == nil { |
Mohamed Abukar | 775722c | 2019-06-10 16:41:57 +0300 | [diff] [blame] | 188 | Logger.Error("rmrClient: Allocating message buffer failed!") |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 189 | } |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 190 | return buf |
| 191 | } |
| 192 | |
Mohamed Abukar | 0213960 | 2019-09-30 08:55:35 +0300 | [diff] [blame] | 193 | func (m *RMRClient) Free(mbuf *C.rmr_mbuf_t) { |
| 194 | if mbuf == nil { |
Mohamed Abukar | 0213960 | 2019-09-30 08:55:35 +0300 | [diff] [blame] | 195 | return |
| 196 | } |
| 197 | C.rmr_free_msg(mbuf) |
| 198 | } |
| 199 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 200 | func (m *RMRClient) SendMsg(params *RMRParams) bool { |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 201 | return m.Send(params, false) |
Mohamed Abukar | 09d0a37 | 2019-07-14 08:45:30 +0300 | [diff] [blame] | 202 | } |
| 203 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 204 | func (m *RMRClient) SendRts(params *RMRParams) bool { |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 205 | return m.Send(params, true) |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 206 | } |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 207 | |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 208 | func (m *RMRClient) Send(params *RMRParams, isRts bool) bool { |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 209 | txBuffer := params.Mbuf |
| 210 | if txBuffer == nil { |
| 211 | txBuffer = m.Allocate() |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 212 | } |
| 213 | |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 214 | txBuffer.mtype = C.int(params.Mtype) |
| 215 | txBuffer.sub_id = C.int(params.SubId) |
| 216 | txBuffer.len = C.int(len(params.Payload)) |
Mohamed Abukar | 688e6c9 | 2019-09-05 13:12:42 +0300 | [diff] [blame] | 217 | if params.PayloadLen != 0 { |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 218 | txBuffer.len = C.int(params.PayloadLen) |
Mohamed Abukar | 688e6c9 | 2019-09-05 13:12:42 +0300 | [diff] [blame] | 219 | } |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 220 | datap := C.CBytes(params.Payload) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 221 | defer C.free(datap) |
| 222 | |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 223 | if params != nil { |
| 224 | if params.Meid != nil { |
| 225 | b := make([]byte, int(C.RMR_MAX_MEID)) |
Mohamed Abukar | 19461e1 | 2019-08-23 08:46:11 +0300 | [diff] [blame] | 226 | copy(b, []byte(params.Meid.RanName)) |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 227 | C.rmr_bytes2meid(txBuffer, (*C.uchar)(unsafe.Pointer(&b[0])), C.int(len(b))) |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 228 | } |
| 229 | xidLen := len(params.Xid) |
| 230 | if xidLen > 0 && xidLen <= C.RMR_MAX_XID { |
Juha Hyttinen | 6e075ce | 2019-11-06 08:42:34 +0200 | [diff] [blame] | 231 | b := make([]byte, int(C.RMR_MAX_XID)) |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 232 | copy(b, []byte(params.Xid)) |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 233 | C.rmr_bytes2xact(txBuffer, (*C.uchar)(unsafe.Pointer(&b[0])), C.int(len(b))) |
Mohamed Abukar | f11ab7a | 2019-08-14 16:55:01 +0300 | [diff] [blame] | 234 | } |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 235 | } |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 236 | C.write_bytes_array(txBuffer.payload, datap, txBuffer.len) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 237 | |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 238 | return m.SendBuf(txBuffer, isRts) |
| 239 | } |
| 240 | |
| 241 | func (m *RMRClient) SendBuf(txBuffer *C.rmr_mbuf_t, isRts bool) bool { |
| 242 | var ( |
| 243 | currBuffer *C.rmr_mbuf_t |
| 244 | state bool = true |
| 245 | counterName string = "Transmitted" |
| 246 | ) |
| 247 | |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 248 | txBuffer.state = 0 |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 249 | if isRts { |
| 250 | currBuffer = C.rmr_rts_msg(m.context, txBuffer) |
| 251 | } else { |
| 252 | currBuffer = C.rmr_send_msg(m.context, txBuffer) |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 253 | } |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 254 | |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 255 | if currBuffer == nil { |
| 256 | m.UpdateStatCounter("TransmitError") |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 257 | return m.LogMBufError("SendBuf failed", txBuffer) |
Mohamed Abukar | f0ee2c6 | 2019-10-14 19:29:24 +0300 | [diff] [blame] | 258 | } |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 259 | |
| 260 | // Just quick retry seems to help for K8s issue |
| 261 | for j := 0; j < 3 && currBuffer != nil && currBuffer.state == C.RMR_ERR_RETRY; j++ { |
| 262 | if isRts { |
| 263 | currBuffer = C.rmr_rts_msg(m.context, currBuffer) |
| 264 | } else { |
| 265 | currBuffer = C.rmr_send_msg(m.context, currBuffer) |
| 266 | } |
| 267 | } |
| 268 | |
| 269 | if currBuffer.state != C.RMR_OK { |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 270 | counterName = "TransmitError" |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 271 | state = m.LogMBufError("SendBuf failed", currBuffer) |
Mohamed Abukar | d5dc8e1 | 2019-10-15 09:58:17 +0300 | [diff] [blame] | 272 | } |
| 273 | |
| 274 | m.UpdateStatCounter(counterName) |
| 275 | m.Free(currBuffer) |
| 276 | return state |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 277 | } |
| 278 | |
| 279 | func (m *RMRClient) UpdateStatCounter(name string) { |
| 280 | m.mux.Lock() |
| 281 | m.stat[name].Inc() |
| 282 | m.mux.Unlock() |
| 283 | } |
| 284 | |
| 285 | func (m *RMRClient) RegisterMetrics() { |
| 286 | m.stat = Metric.RegisterCounterGroup(RMRCounterOpts, "RMR") |
| 287 | } |
| 288 | |
Mohamed Abukar | 2e78e42 | 2019-06-02 11:45:52 +0300 | [diff] [blame] | 289 | func (m *RMRClient) Wait() { |
| 290 | m.wg.Wait() |
| 291 | } |
| 292 | |
| 293 | func (m *RMRClient) IsReady() bool { |
| 294 | return m.ready != 0 |
| 295 | } |
| 296 | |
Mohamed Abukar | 192518d | 2019-06-11 18:06:50 +0300 | [diff] [blame] | 297 | func (m *RMRClient) SetReadyCB(cb ReadyCB, params interface{}) { |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 298 | m.readyCb = cb |
Mohamed Abukar | 192518d | 2019-06-11 18:06:50 +0300 | [diff] [blame] | 299 | m.readyCbParams = params |
Mohamed Abukar | 349a098 | 2019-06-08 18:15:42 +0300 | [diff] [blame] | 300 | } |
Mohamed Abukar | 775722c | 2019-06-10 16:41:57 +0300 | [diff] [blame] | 301 | |
| 302 | func (m *RMRClient) GetRicMessageId(name string) (int, bool) { |
| 303 | id, ok := RICMessageTypes[name] |
| 304 | return id, ok |
| 305 | } |
| 306 | |
| 307 | func (m *RMRClient) GetRicMessageName(id int) (s string) { |
| 308 | for k, v := range RICMessageTypes { |
| 309 | if id == v { |
| 310 | return k |
| 311 | } |
| 312 | } |
| 313 | return |
| 314 | } |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 315 | |
Mohamed Abukar | 467a99f | 2019-11-20 16:07:49 +0200 | [diff] [blame] | 316 | func (m *RMRClient) LogMBufError(text string, mbuf *C.rmr_mbuf_t) bool { |
| 317 | Logger.Debug(fmt.Sprintf("rmrClient: %s -> [tp=%v] %v - %s", text, mbuf.tp_state, mbuf.state, RMRErrors[int(mbuf.state)])) |
| 318 | return false |
| 319 | } |
| 320 | |
Mohamed Abukar | d969b20 | 2019-07-01 17:14:44 +0300 | [diff] [blame] | 321 | // To be removed ... |
| 322 | func (m *RMRClient) GetStat() (r RMRStatistics) { |
| 323 | return |
| 324 | } |