Roger Meier | 6cf0ffc | 2014-04-05 00:45:42 +0200 | [diff] [blame] | 1 | -- |
| 2 | -- Licensed to the Apache Software Foundation (ASF) under one |
| 3 | -- or more contributor license agreements. See the NOTICE file |
| 4 | -- distributed with this work for additional information |
| 5 | -- regarding copyright ownership. The ASF licenses this file |
| 6 | -- to you under the Apache License, Version 2.0 (the |
| 7 | -- "License"); you may not use this file except in compliance |
| 8 | -- with the License. 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, |
| 13 | -- software distributed under the License is distributed on an |
| 14 | -- "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| 15 | -- KIND, either express or implied. See the License for the |
| 16 | -- specific language governing permissions and limitations |
| 17 | -- under the License. |
| 18 | -- |
| 19 | |
| 20 | require 'TProtocol' |
| 21 | require 'libluabpack' |
| 22 | require 'libluabitwise' |
| 23 | |
| 24 | TBinaryProtocol = __TObject.new(TProtocolBase, { |
| 25 | __type = 'TBinaryProtocol', |
| 26 | VERSION_MASK = -65536, -- 0xffff0000 |
| 27 | VERSION_1 = -2147418112, -- 0x80010000 |
| 28 | TYPE_MASK = 0x000000ff, |
| 29 | strictRead = false, |
| 30 | strictWrite = true |
| 31 | }) |
| 32 | |
| 33 | function TBinaryProtocol:writeMessageBegin(name, ttype, seqid) |
Phillipp Röll | 42b9be1 | 2015-06-21 14:38:31 +0200 | [diff] [blame^] | 34 | if self.strictWrite then |
Roger Meier | 6cf0ffc | 2014-04-05 00:45:42 +0200 | [diff] [blame] | 35 | self:writeI32(libluabitwise.bor(TBinaryProtocol.VERSION_1, ttype)) |
| 36 | self:writeString(name) |
| 37 | self:writeI32(seqid) |
| 38 | else |
| 39 | self:writeString(name) |
| 40 | self:writeByte(ttype) |
| 41 | self:writeI32(seqid) |
| 42 | end |
| 43 | end |
| 44 | |
| 45 | function TBinaryProtocol:writeMessageEnd() |
| 46 | end |
| 47 | |
| 48 | function TBinaryProtocol:writeStructBegin(name) |
| 49 | end |
| 50 | |
| 51 | function TBinaryProtocol:writeStructEnd() |
| 52 | end |
| 53 | |
| 54 | function TBinaryProtocol:writeFieldBegin(name, ttype, id) |
| 55 | self:writeByte(ttype) |
| 56 | self:writeI16(id) |
| 57 | end |
| 58 | |
| 59 | function TBinaryProtocol:writeFieldEnd() |
| 60 | end |
| 61 | |
| 62 | function TBinaryProtocol:writeFieldStop() |
| 63 | self:writeByte(TType.STOP); |
| 64 | end |
| 65 | |
| 66 | function TBinaryProtocol:writeMapBegin(ktype, vtype, size) |
| 67 | self:writeByte(ktype) |
| 68 | self:writeByte(vtype) |
| 69 | self:writeI32(size) |
| 70 | end |
| 71 | |
| 72 | function TBinaryProtocol:writeMapEnd() |
| 73 | end |
| 74 | |
| 75 | function TBinaryProtocol:writeListBegin(etype, size) |
| 76 | self:writeByte(etype) |
| 77 | self:writeI32(size) |
| 78 | end |
| 79 | |
| 80 | function TBinaryProtocol:writeListEnd() |
| 81 | end |
| 82 | |
| 83 | function TBinaryProtocol:writeSetBegin(etype, size) |
| 84 | self:writeByte(etype) |
| 85 | self:writeI32(size) |
| 86 | end |
| 87 | |
| 88 | function TBinaryProtocol:writeSetEnd() |
| 89 | end |
| 90 | |
| 91 | function TBinaryProtocol:writeBool(bool) |
| 92 | if bool then |
| 93 | self:writeByte(1) |
| 94 | else |
| 95 | self:writeByte(0) |
| 96 | end |
| 97 | end |
| 98 | |
| 99 | function TBinaryProtocol:writeByte(byte) |
| 100 | local buff = libluabpack.bpack('c', byte) |
| 101 | self.trans:write(buff) |
| 102 | end |
| 103 | |
| 104 | function TBinaryProtocol:writeI16(i16) |
| 105 | local buff = libluabpack.bpack('s', i16) |
| 106 | self.trans:write(buff) |
| 107 | end |
| 108 | |
| 109 | function TBinaryProtocol:writeI32(i32) |
| 110 | local buff = libluabpack.bpack('i', i32) |
| 111 | self.trans:write(buff) |
| 112 | end |
| 113 | |
| 114 | function TBinaryProtocol:writeI64(i64) |
| 115 | local buff = libluabpack.bpack('l', i64) |
| 116 | self.trans:write(buff) |
| 117 | end |
| 118 | |
| 119 | function TBinaryProtocol:writeDouble(dub) |
| 120 | local buff = libluabpack.bpack('d', dub) |
| 121 | self.trans:write(buff) |
| 122 | end |
| 123 | |
| 124 | function TBinaryProtocol:writeString(str) |
| 125 | -- Should be utf-8 |
| 126 | self:writeI32(string.len(str)) |
| 127 | self.trans:write(str) |
| 128 | end |
| 129 | |
| 130 | function TBinaryProtocol:readMessageBegin() |
| 131 | local sz, ttype, name, seqid = self:readI32() |
| 132 | if sz < 0 then |
| 133 | local version = libluabitwise.band(sz, TBinaryProtocol.VERSION_MASK) |
| 134 | if version ~= TBinaryProtocol.VERSION_1 then |
| 135 | terror(TProtocolException:new{ |
| 136 | message = 'Bad version in readMessageBegin: ' .. sz |
| 137 | }) |
| 138 | end |
| 139 | ttype = libluabitwise.band(sz, TBinaryProtocol.TYPE_MASK) |
| 140 | name = self:readString() |
| 141 | seqid = self:readI32() |
| 142 | else |
| 143 | if self.strictRead then |
| 144 | terror(TProtocolException:new{message = 'No protocol version header'}) |
| 145 | end |
| 146 | name = self.trans:readAll(sz) |
| 147 | ttype = self:readByte() |
| 148 | seqid = self:readI32() |
| 149 | end |
| 150 | return name, ttype, seqid |
| 151 | end |
| 152 | |
| 153 | function TBinaryProtocol:readMessageEnd() |
| 154 | end |
| 155 | |
| 156 | function TBinaryProtocol:readStructBegin() |
| 157 | return nil |
| 158 | end |
| 159 | |
| 160 | function TBinaryProtocol:readStructEnd() |
| 161 | end |
| 162 | |
| 163 | function TBinaryProtocol:readFieldBegin() |
| 164 | local ttype = self:readByte() |
| 165 | if ttype == TType.STOP then |
| 166 | return nil, ttype, 0 |
| 167 | end |
| 168 | local id = self:readI16() |
| 169 | return nil, ttype, id |
| 170 | end |
| 171 | |
| 172 | function TBinaryProtocol:readFieldEnd() |
| 173 | end |
| 174 | |
| 175 | function TBinaryProtocol:readMapBegin() |
| 176 | local ktype = self:readByte() |
| 177 | local vtype = self:readByte() |
| 178 | local size = self:readI32() |
| 179 | return ktype, vtype, size |
| 180 | end |
| 181 | |
| 182 | function TBinaryProtocol:readMapEnd() |
| 183 | end |
| 184 | |
| 185 | function TBinaryProtocol:readListBegin() |
| 186 | local etype = self:readByte() |
| 187 | local size = self:readI32() |
| 188 | return etype, size |
| 189 | end |
| 190 | |
| 191 | function TBinaryProtocol:readListEnd() |
| 192 | end |
| 193 | |
| 194 | function TBinaryProtocol:readSetBegin() |
| 195 | local etype = self:readByte() |
| 196 | local size = self:readI32() |
| 197 | return etype, size |
| 198 | end |
| 199 | |
| 200 | function TBinaryProtocol:readSetEnd() |
| 201 | end |
| 202 | |
| 203 | function TBinaryProtocol:readBool() |
| 204 | local byte = self:readByte() |
| 205 | if byte == 0 then |
| 206 | return false |
| 207 | end |
| 208 | return true |
| 209 | end |
| 210 | |
| 211 | function TBinaryProtocol:readByte() |
| 212 | local buff = self.trans:readAll(1) |
| 213 | local val = libluabpack.bunpack('c', buff) |
| 214 | return val |
| 215 | end |
| 216 | |
| 217 | function TBinaryProtocol:readI16() |
| 218 | local buff = self.trans:readAll(2) |
| 219 | local val = libluabpack.bunpack('s', buff) |
| 220 | return val |
| 221 | end |
| 222 | |
| 223 | function TBinaryProtocol:readI32() |
| 224 | local buff = self.trans:readAll(4) |
| 225 | local val = libluabpack.bunpack('i', buff) |
| 226 | return val |
| 227 | end |
| 228 | |
| 229 | function TBinaryProtocol:readI64() |
| 230 | local buff = self.trans:readAll(8) |
| 231 | local val = libluabpack.bunpack('l', buff) |
| 232 | return val |
| 233 | end |
| 234 | |
| 235 | function TBinaryProtocol:readDouble() |
| 236 | local buff = self.trans:readAll(8) |
| 237 | local val = libluabpack.bunpack('d', buff) |
| 238 | return val |
| 239 | end |
| 240 | |
| 241 | function TBinaryProtocol:readString() |
| 242 | local len = self:readI32() |
| 243 | local str = self.trans:readAll(len) |
| 244 | return str |
| 245 | end |
| 246 | |
| 247 | TBinaryProtocolFactory = TProtocolFactory:new{ |
| 248 | __type = 'TBinaryProtocolFactory', |
| 249 | strictRead = false |
| 250 | } |
| 251 | |
| 252 | function TBinaryProtocolFactory:getProtocol(trans) |
| 253 | -- TODO Enforce that this must be a transport class (ie not a bool) |
| 254 | if not trans then |
| 255 | terror(TProtocolException:new{ |
| 256 | message = 'Must supply a transport to ' .. ttype(self) |
| 257 | }) |
| 258 | end |
| 259 | return TBinaryProtocol:new{ |
| 260 | trans = trans, |
| 261 | strictRead = self.strictRead, |
| 262 | strictWrite = true |
| 263 | } |
| 264 | end |