| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 1 | // Go support for Protocol Buffers - Google's data interchange format |
| 2 | // |
| 3 | // Copyright 2010 Google Inc. All rights reserved. |
| 4 | // http://code.google.com/p/goprotobuf/ |
| 5 | // |
| 6 | // Redistribution and use in source and binary forms, with or without |
| 7 | // modification, are permitted provided that the following conditions are |
| 8 | // met: |
| 9 | // |
| 10 | // * Redistributions of source code must retain the above copyright |
| 11 | // notice, this list of conditions and the following disclaimer. |
| 12 | // * Redistributions in binary form must reproduce the above |
| 13 | // copyright notice, this list of conditions and the following disclaimer |
| 14 | // in the documentation and/or other materials provided with the |
| 15 | // distribution. |
| 16 | // * Neither the name of Google Inc. nor the names of its |
| 17 | // contributors may be used to endorse or promote products derived from |
| 18 | // this software without specific prior written permission. |
| 19 | // |
| 20 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 21 | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 22 | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 23 | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 24 | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 25 | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 26 | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 27 | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 28 | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 29 | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 30 | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 31 | |
| 32 | package proto_test |
| 33 | |
| 34 | import ( |
| 35 | "bytes" |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 36 | "strings" |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 37 | "testing" |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 38 | |
| Rob Pike | 3f6f2d8 | 2011-12-18 13:55:35 -0800 | [diff] [blame] | 39 | "code.google.com/p/goprotobuf/proto" |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 40 | |
| David Symonds | 704096f | 2012-03-15 15:10:26 +1100 | [diff] [blame] | 41 | pb "./testdata" |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 42 | ) |
| 43 | |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 44 | func newTestMessage() *pb.MyMessage { |
| 45 | msg := &pb.MyMessage{ |
| 46 | Count: proto.Int32(42), |
| 47 | Name: proto.String("Dave"), |
| 48 | Quote: proto.String(`"I didn't want to go."`), |
| Rob Pike | 9caa5b9 | 2010-05-11 16:04:57 -0700 | [diff] [blame] | 49 | Pet: []string{"bunny", "kitty", "horsey"}, |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 50 | Inner: &pb.InnerMessage{ |
| 51 | Host: proto.String("footrest.syd"), |
| 52 | Port: proto.Int32(7001), |
| 53 | Connected: proto.Bool(true), |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 54 | }, |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 55 | Others: []*pb.OtherMessage{ |
| David Symonds | 92dd6c1 | 2012-03-23 10:59:49 +1100 | [diff] [blame] | 56 | { |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 57 | Key: proto.Int64(0xdeadbeef), |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 58 | Value: []byte{1, 65, 7, 12}, |
| 59 | }, |
| David Symonds | 92dd6c1 | 2012-03-23 10:59:49 +1100 | [diff] [blame] | 60 | { |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 61 | Weight: proto.Float32(6.022), |
| 62 | Inner: &pb.InnerMessage{ |
| 63 | Host: proto.String("lesha.mtv"), |
| 64 | Port: proto.Int32(8002), |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 65 | }, |
| 66 | }, |
| 67 | }, |
| David Symonds | efeca9a | 2012-05-08 10:36:04 +1000 | [diff] [blame] | 68 | Bikeshed: pb.MyMessage_BLUE.Enum(), |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 69 | Somegroup: &pb.MyMessage_SomeGroup{ |
| 70 | GroupField: proto.Int32(8), |
| David Symonds | 9f40281 | 2011-04-28 18:08:44 +1000 | [diff] [blame] | 71 | }, |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 72 | // One normally wouldn't do this. |
| 73 | // This is an undeclared tag 13, as a varint (wire type 0) with value 4. |
| 74 | XXX_unrecognized: []byte{13<<3 | 0, 4}, |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 75 | } |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 76 | ext := &pb.Ext{ |
| 77 | Data: proto.String("Big gobs for big rats"), |
| David Symonds | e37856c | 2011-06-22 12:52:53 +1000 | [diff] [blame] | 78 | } |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 79 | if err := proto.SetExtension(msg, pb.E_Ext_More, ext); err != nil { |
| David Symonds | e37856c | 2011-06-22 12:52:53 +1000 | [diff] [blame] | 80 | panic(err) |
| 81 | } |
| David Symonds | 61826da | 2012-05-05 09:31:28 +1000 | [diff] [blame] | 82 | greetings := []string{"adg", "easy", "cow"} |
| 83 | if err := proto.SetExtension(msg, pb.E_Greeting, greetings); err != nil { |
| 84 | panic(err) |
| 85 | } |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 86 | |
| 87 | // Add an unknown extension. We marshal a pb.Ext, and fake the ID. |
| 88 | b, err := proto.Marshal(&pb.Ext{Data: proto.String("3G skiing")}) |
| 89 | if err != nil { |
| 90 | panic(err) |
| 91 | } |
| David Symonds | 5453105 | 2011-12-08 12:00:31 +1100 | [diff] [blame] | 92 | b = append(proto.EncodeVarint(201<<3|proto.WireBytes), b...) |
| 93 | proto.SetRawExtension(msg, 201, b) |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 94 | |
| 95 | // Extensions can be plain fields, too, so let's test that. |
| David Symonds | 5453105 | 2011-12-08 12:00:31 +1100 | [diff] [blame] | 96 | b = append(proto.EncodeVarint(202<<3|proto.WireVarint), 19) |
| 97 | proto.SetRawExtension(msg, 202, b) |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 98 | |
| David Symonds | e37856c | 2011-06-22 12:52:53 +1000 | [diff] [blame] | 99 | return msg |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 100 | } |
| 101 | |
| 102 | const text = `count: 42 |
| 103 | name: "Dave" |
| 104 | quote: "\"I didn't want to go.\"" |
| 105 | pet: "bunny" |
| 106 | pet: "kitty" |
| 107 | pet: "horsey" |
| 108 | inner: < |
| 109 | host: "footrest.syd" |
| 110 | port: 7001 |
| 111 | connected: true |
| 112 | > |
| 113 | others: < |
| 114 | key: 3735928559 |
| David Symonds | 4c95bfe | 2011-09-13 14:43:27 +1000 | [diff] [blame] | 115 | value: "\001A\007\014" |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 116 | > |
| 117 | others: < |
| 118 | weight: 6.022 |
| 119 | inner: < |
| 120 | host: "lesha.mtv" |
| 121 | port: 8002 |
| 122 | > |
| 123 | > |
| 124 | bikeshed: BLUE |
| David Symonds | 9f40281 | 2011-04-28 18:08:44 +1000 | [diff] [blame] | 125 | SomeGroup { |
| 126 | group_field: 8 |
| 127 | } |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 128 | /* 2 unknown bytes */ |
| 129 | tag13: 4 |
| Rob Pike | b7907bf | 2012-02-13 14:25:20 +1100 | [diff] [blame] | 130 | [testdata.Ext.more]: < |
| David Symonds | e37856c | 2011-06-22 12:52:53 +1000 | [diff] [blame] | 131 | data: "Big gobs for big rats" |
| 132 | > |
| David Symonds | 61826da | 2012-05-05 09:31:28 +1000 | [diff] [blame] | 133 | [testdata.greeting]: "adg" |
| 134 | [testdata.greeting]: "easy" |
| 135 | [testdata.greeting]: "cow" |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 136 | /* 13 unknown bytes */ |
| David Symonds | 5453105 | 2011-12-08 12:00:31 +1100 | [diff] [blame] | 137 | tag201: "\t3G skiing" |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 138 | /* 3 unknown bytes */ |
| David Symonds | 5453105 | 2011-12-08 12:00:31 +1100 | [diff] [blame] | 139 | tag202: 19 |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 140 | ` |
| 141 | |
| 142 | func TestMarshalTextFull(t *testing.T) { |
| 143 | buf := new(bytes.Buffer) |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 144 | proto.MarshalText(buf, newTestMessage()) |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 145 | s := buf.String() |
| 146 | if s != text { |
| 147 | t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", s, text) |
| 148 | } |
| 149 | } |
| 150 | |
| David Symonds | dbdc421 | 2012-11-08 08:20:35 +1100 | [diff] [blame^] | 151 | func BenchmarkMarshalTextFull(b *testing.B) { |
| 152 | buf := new(bytes.Buffer) |
| 153 | m := newTestMessage() |
| 154 | for i := 0; i < b.N; i++ { |
| 155 | buf.Reset() |
| 156 | proto.MarshalText(buf, m) |
| 157 | } |
| 158 | } |
| 159 | |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 160 | func compact(src string) string { |
| David Symonds | e37856c | 2011-06-22 12:52:53 +1000 | [diff] [blame] | 161 | // s/[ \n]+/ /g; s/ $//; |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 162 | dst := make([]byte, len(src)) |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 163 | space, comment := false, false |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 164 | j := 0 |
| 165 | for i := 0; i < len(src); i++ { |
| David Symonds | 1d72f7a | 2011-08-19 18:28:52 +1000 | [diff] [blame] | 166 | if strings.HasPrefix(src[i:], "/*") { |
| 167 | comment = true |
| 168 | i++ |
| 169 | continue |
| 170 | } |
| 171 | if comment && strings.HasPrefix(src[i:], "*/") { |
| 172 | comment = false |
| 173 | i++ |
| 174 | continue |
| 175 | } |
| 176 | if comment { |
| 177 | continue |
| 178 | } |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 179 | c := src[i] |
| 180 | if c == ' ' || c == '\n' { |
| 181 | space = true |
| 182 | continue |
| 183 | } |
| David Symonds | 9f40281 | 2011-04-28 18:08:44 +1000 | [diff] [blame] | 184 | if j > 0 && (dst[j-1] == ':' || dst[j-1] == '<' || dst[j-1] == '{') { |
| 185 | space = false |
| 186 | } |
| 187 | if c == '{' { |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 188 | space = false |
| 189 | } |
| 190 | if space { |
| 191 | dst[j] = ' ' |
| 192 | j++ |
| 193 | space = false |
| 194 | } |
| 195 | dst[j] = c |
| 196 | j++ |
| 197 | } |
| 198 | if space { |
| 199 | dst[j] = ' ' |
| 200 | j++ |
| 201 | } |
| 202 | return string(dst[0:j]) |
| 203 | } |
| 204 | |
| 205 | var compactText = compact(text) |
| 206 | |
| 207 | func TestCompactText(t *testing.T) { |
| David Symonds | aa922ff | 2011-07-19 14:58:06 +1000 | [diff] [blame] | 208 | s := proto.CompactTextString(newTestMessage()) |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 209 | if s != compactText { |
| David Symonds | 4c95bfe | 2011-09-13 14:43:27 +1000 | [diff] [blame] | 210 | t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v\n===\n", s, compactText) |
| 211 | } |
| 212 | } |
| 213 | |
| 214 | func TestStringEscaping(t *testing.T) { |
| 215 | testCases := []struct { |
| 216 | in *pb.Strings |
| 217 | out string |
| 218 | }{ |
| 219 | { |
| 220 | // Test data from C++ test (TextFormatTest.StringEscape). |
| 221 | // Single divergence: we don't escape apostrophes. |
| 222 | &pb.Strings{StringField: proto.String("\"A string with ' characters \n and \r newlines and \t tabs and \001 slashes \\ and multiple spaces")}, |
| 223 | "string_field: \"\\\"A string with ' characters \\n and \\r newlines and \\t tabs and \\001 slashes \\\\ and multiple spaces\"\n", |
| 224 | }, |
| 225 | { |
| 226 | // Test data from the same C++ test. |
| 227 | &pb.Strings{StringField: proto.String("\350\260\267\346\255\214")}, |
| 228 | "string_field: \"\\350\\260\\267\\346\\255\\214\"\n", |
| 229 | }, |
| David Symonds | fa94a1e | 2012-09-24 13:21:49 +1000 | [diff] [blame] | 230 | { |
| 231 | // Some UTF-8. |
| 232 | &pb.Strings{StringField: proto.String("\x00\x01\xff\x81")}, |
| 233 | `string_field: "\000\001\377\201"` + "\n", |
| 234 | }, |
| David Symonds | 4c95bfe | 2011-09-13 14:43:27 +1000 | [diff] [blame] | 235 | } |
| 236 | |
| 237 | for i, tc := range testCases { |
| 238 | var buf bytes.Buffer |
| 239 | proto.MarshalText(&buf, tc.in) |
| David Symonds | fa94a1e | 2012-09-24 13:21:49 +1000 | [diff] [blame] | 240 | s := buf.String() |
| 241 | if s != tc.out { |
| David Symonds | 4c95bfe | 2011-09-13 14:43:27 +1000 | [diff] [blame] | 242 | t.Errorf("#%d: Got:\n%s\nExpected:\n%s\n", i, s, tc.out) |
| David Symonds | fa94a1e | 2012-09-24 13:21:49 +1000 | [diff] [blame] | 243 | continue |
| 244 | } |
| 245 | |
| 246 | // Check round-trip. |
| 247 | pb := new(pb.Strings) |
| 248 | if err := proto.UnmarshalText(s, pb); err != nil { |
| 249 | t.Errorf("#%d: UnmarshalText: %v", i, err) |
| 250 | continue |
| 251 | } |
| 252 | if !proto.Equal(pb, tc.in) { |
| 253 | t.Errorf("#%d: Round-trip failed:\nstart: %v\n end: %v", i, tc.in, pb) |
| David Symonds | 4c95bfe | 2011-09-13 14:43:27 +1000 | [diff] [blame] | 254 | } |
| Rob Pike | aaa3a62 | 2010-03-20 22:32:34 -0700 | [diff] [blame] | 255 | } |
| 256 | } |