blob: d667ec2b3e92536ca24dfbdffb134d3d876aa591 [file] [log] [blame]
Daniel Jasper7052ce62014-01-19 09:04:08 +00001//===- unittest/Format/FormatTestProto.cpp --------------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Daniel Jasper7052ce62014-01-19 09:04:08 +000010#include "FormatTestUtils.h"
11#include "clang/Format/Format.h"
12#include "llvm/Support/Debug.h"
13#include "gtest/gtest.h"
14
Chandler Carruth10346662014-04-22 03:17:02 +000015#define DEBUG_TYPE "format-test"
16
Daniel Jasper7052ce62014-01-19 09:04:08 +000017namespace clang {
18namespace format {
19
20class FormatTestProto : public ::testing::Test {
21protected:
22 static std::string format(llvm::StringRef Code, unsigned Offset,
23 unsigned Length, const FormatStyle &Style) {
24 DEBUG(llvm::errs() << "---\n");
25 DEBUG(llvm::errs() << Code << "\n\n");
26 std::vector<tooling::Range> Ranges(1, tooling::Range(Offset, Length));
27 tooling::Replacements Replaces = reformat(Style, Code, Ranges);
Eric Liu4f8d9942016-07-11 13:53:12 +000028 auto Result = applyAllReplacements(Code, Replaces);
29 EXPECT_TRUE(static_cast<bool>(Result));
30 DEBUG(llvm::errs() << "\n" << *Result << "\n\n");
31 return *Result;
Daniel Jasper7052ce62014-01-19 09:04:08 +000032 }
33
34 static std::string format(llvm::StringRef Code) {
Nico Weber514ecc82014-02-02 20:50:45 +000035 FormatStyle Style = getGoogleStyle(FormatStyle::LK_Proto);
Daniel Jasper215d6c82014-01-22 08:04:52 +000036 Style.ColumnLimit = 60; // To make writing tests easier.
37 return format(Code, 0, Code.size(), Style);
Daniel Jasper7052ce62014-01-19 09:04:08 +000038 }
39
40 static void verifyFormat(llvm::StringRef Code) {
41 EXPECT_EQ(Code.str(), format(test::messUp(Code)));
42 }
43};
44
45TEST_F(FormatTestProto, FormatsMessages) {
46 verifyFormat("message SomeMessage {\n"
47 " required int32 field1 = 1;\n"
48 "}");
49 verifyFormat("message SomeMessage {\n"
Daniel Jasper9c2820c2014-06-23 07:36:25 +000050 " required .absolute.Reference field1 = 1;\n"
51 "}");
52 verifyFormat("message SomeMessage {\n"
Daniel Jasper7052ce62014-01-19 09:04:08 +000053 " required int32 field1 = 1;\n"
54 " optional string field2 = 2 [default = \"2\"]\n"
55 "}");
Daniel Jasper215d6c82014-01-22 08:04:52 +000056
57 verifyFormat("message SomeMessage {\n"
Nikola Smiljanice08a91e2014-05-08 00:05:13 +000058 " optional really.really.long.qualified.type.aaa.aaaaaaa\n"
Daniel Jasper215d6c82014-01-22 08:04:52 +000059 " fiiiiiiiiiiiiiiiiiiiiiiiiield = 1;\n"
60 " optional\n"
Nikola Smiljanice08a91e2014-05-08 00:05:13 +000061 " really.really.long.qualified.type.aaa.aaaaaaa.aaaaaaaa\n"
Daniel Jasper215d6c82014-01-22 08:04:52 +000062 " another_fiiiiiiiiiiiiiiiiiiiiield = 2;\n"
63 "}");
Krasimir Georgiev9451e672017-06-27 10:06:40 +000064 verifyFormat("message SomeMessage {\n"
65 " map<string, Project> projects = 1;\n"
66 " optional map<string, int32> size_projects = 2;\n"
67 " map<int, really.really.really.long.qualified.type.nameeee>\n"
68 " projects = 3;\n"
69 " map<int, really.really.really.really.long.qualified.type\n"
70 " .nameeee> projects = 4;\n"
71 " map<int,\n"
72 " reallyreallyreallyreallyreallyreallyreallylongname>\n"
73 " projects = 5;\n"
74 " map<int, Project>\n"
75 " longlonglonglonglonglonglonglonglonglongonglon = 6;\n"
76 " map<releleallyreallyreallyreallyreallyreallyreallylongname,\n"
77 " int> projects = 7;\n"
78 " map<releleallyreallyreallyreallyreallyreallyreallylongname,\n"
79 " releleallyreallyreallyreallyreallyreallyreallylongname>\n"
80 " releleallyreallyreallyreallyreallyreallyreallylongnam =\n"
81 " 8;\n"
82 " map<relele.llyreal.yreallyr.allyreally.eallyreal\n"
83 " .sauenirylongname,\n"
84 " really.really.really.really.long.qualified.type\n"
85 " .nameeee> projects = 9;\n"
86 "}");
Daniel Jasper7052ce62014-01-19 09:04:08 +000087}
88
Daniel Jasper23d3bcf2015-06-25 08:38:46 +000089TEST_F(FormatTestProto, KeywordsInOtherLanguages) {
90 verifyFormat("optional string operator = 1;");
91}
92
Daniel Jasper7052ce62014-01-19 09:04:08 +000093TEST_F(FormatTestProto, FormatsEnums) {
94 verifyFormat("enum Type {\n"
95 " UNKNOWN = 0;\n"
96 " TYPE_A = 1;\n"
97 " TYPE_B = 2;\n"
98 "};");
Daniel Jasperc6dd2732015-07-16 14:25:43 +000099 verifyFormat("enum Type {\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000100 " UNKNOWN = 0 [(some_options) = { a: aa, b: bb }];\n"
Daniel Jasper7f3b99f2016-06-23 09:40:19 +0000101 "};");
102 verifyFormat("enum Type {\n"
Daniel Jasperc6dd2732015-07-16 14:25:43 +0000103 " UNKNOWN = 0 [(some_options) = {\n"
Daniel Jasper7f3b99f2016-06-23 09:40:19 +0000104 " a: aa, // wrap\n"
Daniel Jasperc6dd2732015-07-16 14:25:43 +0000105 " b: bb\n"
106 " }];\n"
107 "};");
Daniel Jasper7052ce62014-01-19 09:04:08 +0000108}
109
110TEST_F(FormatTestProto, UnderstandsReturns) {
111 verifyFormat("rpc Search(SearchRequest) returns (SearchResponse);");
112}
113
114TEST_F(FormatTestProto, MessageFieldAttributes) {
115 verifyFormat("optional string test = 1 [default = \"test\"];");
Daniel Jaspera0e9be22014-01-28 18:51:11 +0000116 verifyFormat("optional bool a = 1 [default = true, deprecated = true];");
Daniel Jasperccff4d12016-01-04 07:27:33 +0000117 verifyFormat("optional LongMessageType long_proto_field = 1 [\n"
118 " default = REALLY_REALLY_LONG_CONSTANT_VALUE,\n"
119 " deprecated = true\n"
120 "];");
Daniel Jasper7052ce62014-01-19 09:04:08 +0000121 verifyFormat("optional LongMessageType long_proto_field = 1\n"
122 " [default = REALLY_REALLY_LONG_CONSTANT_VALUE];");
Daniel Jasper6e58fee2014-01-29 18:43:40 +0000123 verifyFormat("repeated double value = 1\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000124 " [(aaaaaaa.aaaaaaaaa) = { aaaaaaaaaaaaaaaaa: AAAAAAAA }];");
Daniel Jasperce2fdb02014-10-27 13:25:59 +0000125 verifyFormat("repeated double value = 1 [(aaaaaaa.aaaaaaaaa) = {\n"
126 " aaaaaaaaaaaaaaaa: AAAAAAAAAA,\n"
127 " bbbbbbbbbbbbbbbb: BBBBBBBBBB\n"
128 "}];");
129 verifyFormat("repeated double value = 1 [(aaaaaaa.aaaaaaaaa) = {\n"
130 " aaaaaaaaaaaaaaaa: AAAAAAAAAA\n"
131 " bbbbbbbbbbbbbbbb: BBBBBBBBBB\n"
132 "}];");
Daniel Jasperccff4d12016-01-04 07:27:33 +0000133 verifyFormat("repeated double value = 1 [\n"
134 " (aaaaaaa.aaaaaaaaa) = {\n"
135 " aaaaaaaaaaaaaaaa: AAAAAAAAAA\n"
136 " bbbbbbbbbbbbbbbb: BBBBBBBBBB\n"
137 " },\n"
138 " (bbbbbbb.bbbbbbbbb) = {\n"
139 " aaaaaaaaaaaaaaaa: AAAAAAAAAA\n"
140 " bbbbbbbbbbbbbbbb: BBBBBBBBBB\n"
141 " }\n"
142 "];");
Daniel Jasperce2fdb02014-10-27 13:25:59 +0000143 verifyFormat("repeated double value = 1 [(aaaaaaa.aaaaaaaaa) = {\n"
Daniel Jasper1662bfe2015-04-03 21:15:46 +0000144 " type: \"AAAAAAAAAA\"\n"
145 " is: \"AAAAAAAAAA\"\n"
146 " or: \"BBBBBBBBBB\"\n"
147 "}];");
148 verifyFormat("repeated double value = 1 [(aaaaaaa.aaaaaaaaa) = {\n"
Daniel Jasperce2fdb02014-10-27 13:25:59 +0000149 " aaaaaaaaaaaaaaaa: AAAAAAAAAA,\n"
150 " bbbbbbb: BBBB,\n"
151 " bbbb: BBB\n"
152 "}];");
Daniel Jasperccff4d12016-01-04 07:27:33 +0000153 verifyFormat("optional AAA aaa = 1 [\n"
154 " foo = {\n"
155 " key: 'a' //\n"
156 " },\n"
157 " bar = {\n"
158 " key: 'a' //\n"
159 " }\n"
160 "];");
Daniel Jasperda910e02017-02-03 08:29:02 +0000161 verifyFormat("optional string test = 1 [default =\n"
162 " \"test\"\n"
163 " \"test\"];");
Daniel Jasperc79e4d22017-03-14 00:40:32 +0000164 verifyFormat("optional Aaaaaaaa aaaaaaaa = 12 [\n"
165 " (aaa) = aaaa,\n"
166 " (bbbbbbbbbbbbbbbbbbbbbbbbbb) = {\n"
167 " aaaaaaaaaaaaaaaaa: true,\n"
168 " aaaaaaaaaaaaaaaa: true\n"
169 " }\n"
170 "];");
Daniel Jasper7052ce62014-01-19 09:04:08 +0000171}
172
Daniel Jasper1b998812015-04-23 09:54:10 +0000173TEST_F(FormatTestProto, DoesntWrapFileOptions) {
174 EXPECT_EQ(
175 "option java_package = "
176 "\"some.really.long.package.that.exceeds.the.column.limit\";",
177 format("option java_package = "
178 "\"some.really.long.package.that.exceeds.the.column.limit\";"));
179}
180
Daniel Jasper929b1db2014-01-20 16:47:22 +0000181TEST_F(FormatTestProto, FormatsOptions) {
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000182 verifyFormat("option (MyProto.options) = {\n"
183 " field_a: OK\n"
184 " field_b: \"OK\"\n"
185 " field_c: \"OK\"\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000186 " msg_field: { field_d: 123 }\n"
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000187 "};");
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000188 verifyFormat("option (MyProto.options) = {\n"
189 " field_a: OK\n"
190 " field_b: \"OK\"\n"
191 " field_c: \"OK\"\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000192 " msg_field: { field_d: 123 field_e: OK }\n"
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000193 "};");
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000194 verifyFormat("option (MyProto.options) = {\n"
195 " field_a: OK // Comment\n"
196 " field_b: \"OK\"\n"
197 " field_c: \"OK\"\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000198 " msg_field: { field_d: 123 }\n"
Daniel Jaspera382cbe2014-07-28 14:08:09 +0000199 "};");
Daniel Jasper497d9fd2014-08-15 05:00:35 +0000200 verifyFormat("option (MyProto.options) = {\n"
201 " field_c: \"OK\"\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000202 " msg_field { field_d: 123 }\n"
Daniel Jasper497d9fd2014-08-15 05:00:35 +0000203 "};");
Krasimir Georgievfeeba612017-06-29 13:30:41 +0000204 verifyFormat("option (MyProto.options) = {\n"
205 " field_a: OK\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000206 " field_b { field_c: OK }\n"
Krasimir Georgievfeeba612017-06-29 13:30:41 +0000207 " field_d: OKOKOK\n"
208 " field_e: OK\n"
209 "}");
Daniel Jasperffbad0e2016-01-04 07:28:12 +0000210
211 // Support syntax with <> instead of {}.
212 verifyFormat("option (MyProto.options) = {\n"
213 " field_c: \"OK\",\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000214 " msg_field: < field_d: 123 >\n"
215 " empty: <>\n"
216 " empty <>\n"
Daniel Jasperffbad0e2016-01-04 07:28:12 +0000217 "};");
Krasimir Georgievff747be2017-06-27 13:43:07 +0000218
219 verifyFormat("option (MyProto.options) = {\n"
Krasimir Georgievfeeba612017-06-29 13:30:41 +0000220 " field_a: OK\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000221 " field_b < field_c: OK >\n"
Krasimir Georgievfeeba612017-06-29 13:30:41 +0000222 " field_d: OKOKOK\n"
223 " field_e: OK\n"
224 "}");
225
226 verifyFormat("option (MyProto.options) = {\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000227 " msg_field: <>\n"
228 " field_c: \"OK\",\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000229 " msg_field: < field_d: 123 >\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000230 " field_e: OK\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000231 " msg_field: < field_d: 12 >\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000232 "};");
233
234 verifyFormat("option (MyProto.options) = <\n"
235 " field_a: OK\n"
236 " field_b: \"OK\"\n"
237 " field_c: 1\n"
238 " field_d: 12.5\n"
239 " field_e: OK\n"
240 ">;");
241
242 verifyFormat("option (MyProto.options) = <\n"
243 " field_a: OK,\n"
244 " field_b: \"OK\",\n"
245 " field_c: 1,\n"
246 " field_d: 12.5,\n"
247 " field_e: OK,\n"
248 ">;");
249
250 verifyFormat("option (MyProto.options) = <\n"
251 " field_a: \"OK\"\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000252 " msg_field: { field_b: OK }\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000253 " field_g: OK\n"
254 " field_g: OK\n"
255 " field_g: OK\n"
256 ">;");
257
258 verifyFormat("option (MyProto.options) = <\n"
259 " field_a: \"OK\"\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000260 " msg_field <\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000261 " field_b: OK\n"
262 " field_c: OK\n"
263 " field_d: OK\n"
264 " field_e: OK\n"
265 " field_f: OK\n"
266 " >\n"
267 " field_g: OK\n"
268 ">;");
269
270 verifyFormat("option (MyProto.options) = <\n"
271 " field_a: \"OK\"\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000272 " msg_field <\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000273 " field_b: OK,\n"
274 " field_c: OK,\n"
275 " field_d: OK,\n"
276 " field_e: OK,\n"
277 " field_f: OK\n"
278 " >\n"
279 " field_g: OK\n"
280 ">;");
281
282 verifyFormat("option (MyProto.options) = <\n"
283 " field_a: \"OK\"\n"
284 " msg_field: <\n"
285 " field_b: OK\n"
286 " field_c: OK\n"
287 " field_d: OK\n"
288 " field_e: OK\n"
289 " field_f: OK\n"
290 " >\n"
291 " field_g: OK\n"
292 ">;");
293
294 verifyFormat("option (MyProto.options) = <\n"
295 " field_a: \"OK\"\n"
296 " msg_field: {\n"
297 " field_b: OK\n"
298 " field_c: OK\n"
299 " field_d: OK\n"
300 " field_e: OK\n"
301 " field_f: OK\n"
302 " }\n"
303 " field_g: OK\n"
304 ">;");
305
306 verifyFormat("option (MyProto.options) = <\n"
307 " field_a: \"OK\"\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000308 " msg_field {\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000309 " field_b: OK\n"
310 " field_c: OK\n"
311 " field_d: OK\n"
312 " field_e: OK\n"
313 " field_f: OK\n"
314 " }\n"
315 " field_g: OK\n"
316 ">;");
317
318 verifyFormat("option (MyProto.options) = {\n"
319 " field_a: \"OK\"\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000320 " msg_field <\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000321 " field_b: OK\n"
322 " field_c: OK\n"
323 " field_d: OK\n"
324 " field_e: OK\n"
325 " field_f: OK\n"
326 " >\n"
327 " field_g: OK\n"
328 "};");
329
330 verifyFormat("option (MyProto.options) = {\n"
331 " field_a: \"OK\"\n"
332 " msg_field: <\n"
333 " field_b: OK\n"
334 " field_c: OK\n"
335 " field_d: OK\n"
336 " field_e: OK\n"
337 " field_f: OK\n"
338 " >\n"
339 " field_g: OK\n"
340 "};");
341
342 verifyFormat("option (MyProto.options) = <\n"
343 " field_a: \"OK\"\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000344 " msg_field {\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000345 " field_b: OK\n"
346 " field_c: OK\n"
347 " field_d: OK\n"
Krasimir Georgiev47f21ef2017-07-06 13:58:29 +0000348 " msg_field <\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000349 " field_A: 1\n"
350 " field_B: 2\n"
351 " field_C: 3\n"
352 " field_D: 4\n"
353 " field_E: 5\n"
354 " >\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000355 " msg_field < field_A: 1 field_B: 2 field_C: 3 f_D: 4 >\n"
Krasimir Georgievff747be2017-06-27 13:43:07 +0000356 " field_e: OK\n"
357 " field_f: OK\n"
358 " }\n"
359 " field_g: OK\n"
360 ">;");
Krasimir Georgievfa4dbb62017-08-03 13:43:45 +0000361
362 verifyFormat("option (MyProto.options) = <\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000363 " data1 < key1: value1 >\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000364 " data2 { key2: value2 }\n"
Krasimir Georgievfa4dbb62017-08-03 13:43:45 +0000365 ">;");
Krasimir Georgievddb19242017-08-03 14:17:29 +0000366
367 verifyFormat("option (MyProto.options) = <\n"
368 " app_id: 'com.javax.swing.salsa.latino'\n"
369 " head_id: 1\n"
Krasimir Georgieva79d62d2018-02-06 11:34:34 +0000370 " data < key: value >\n"
Krasimir Georgievddb19242017-08-03 14:17:29 +0000371 ">;");
372
373 verifyFormat("option (MyProto.options) = {\n"
374 " app_id: 'com.javax.swing.salsa.latino'\n"
375 " head_id: 1\n"
376 " headheadheadheadheadhead_id: 1\n"
Krasimir Georgievc2091802018-01-31 10:14:10 +0000377 " product_data { product { 1 } }\n"
Krasimir Georgievddb19242017-08-03 14:17:29 +0000378 "};");
Daniel Jasper929b1db2014-01-20 16:47:22 +0000379}
380
Daniel Jasper220c0d12014-04-10 07:27:12 +0000381TEST_F(FormatTestProto, FormatsService) {
382 verifyFormat("service SearchService {\n"
383 " rpc Search(SearchRequest) returns (SearchResponse) {\n"
384 " option foo = true;\n"
385 " }\n"
386 "};");
387}
388
Daniel Jasperd8d98392015-11-20 14:32:54 +0000389TEST_F(FormatTestProto, ExtendingMessage) {
390 verifyFormat("extend .foo.Bar {\n"
391 "}");
392}
393
Daniel Jasper3d5a7d62016-06-20 18:20:38 +0000394TEST_F(FormatTestProto, FormatsImports) {
395 verifyFormat("import \"a.proto\";\n"
396 "import \"b.proto\";\n"
397 "// comment\n"
398 "message A {\n"
399 "}");
400
Daniel Jasper8b61d142016-06-20 20:39:53 +0000401 verifyFormat("import public \"a.proto\";\n"
402 "import \"b.proto\";\n"
403 "// comment\n"
404 "message A {\n"
405 "}");
406
Daniel Jasper3d5a7d62016-06-20 18:20:38 +0000407 // Missing semicolons should not confuse clang-format.
408 verifyFormat("import \"a.proto\"\n"
409 "import \"b.proto\"\n"
410 "// comment\n"
411 "message A {\n"
412 "}");
413}
414
Krasimir Georgiev374e6de2018-02-08 10:47:12 +0000415TEST_F(FormatTestProto, KeepsLongStringLiteralsOnSameLine) {
416 verifyFormat(
417 "option (MyProto.options) = {\n"
418 " foo: {\n"
419 " text: \"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaasaaaaaaaa\"\n"
420 " }\n"
421 "}");
422}
423
Alexander Kornienkoab9db512015-06-22 23:07:51 +0000424} // end namespace tooling
Daniel Jasper7052ce62014-01-19 09:04:08 +0000425} // end namespace clang