1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52 package org.eclipse.jgit.lib;
53
54 import static java.nio.charset.StandardCharsets.UTF_8;
55
56 import java.text.MessageFormat;
57 import java.util.ArrayList;
58 import java.util.Collections;
59 import java.util.List;
60 import java.util.Locale;
61 import java.util.Set;
62 import java.util.concurrent.TimeUnit;
63 import java.util.concurrent.atomic.AtomicReference;
64
65 import org.eclipse.jgit.annotations.Nullable;
66 import org.eclipse.jgit.errors.ConfigInvalidException;
67 import org.eclipse.jgit.events.ConfigChangedEvent;
68 import org.eclipse.jgit.events.ConfigChangedListener;
69 import org.eclipse.jgit.events.ListenerHandle;
70 import org.eclipse.jgit.events.ListenerList;
71 import org.eclipse.jgit.internal.JGitText;
72 import org.eclipse.jgit.transport.RefSpec;
73 import org.eclipse.jgit.util.RawParseUtils;
74
75
76
77
78 public class Config {
79
80 private static final String[] EMPTY_STRING_ARRAY = {};
81
82 static final long KiB = 1024;
83 static final long MiB = 1024 * KiB;
84 static final long GiB = 1024 * MiB;
85 private static final int MAX_DEPTH = 10;
86
87 private static final TypedConfigGetter DEFAULT_GETTER = new DefaultTypedConfigGetter();
88
89 private static TypedConfigGetter typedGetter = DEFAULT_GETTER;
90
91
92 private final ListenerList listeners = new ListenerList();
93
94
95
96
97
98
99
100 private final AtomicReference<ConfigSnapshot> state;
101
102 private final Config baseConfig;
103
104
105
106
107
108
109
110
111 static final String MAGIC_EMPTY_VALUE = new String();
112
113
114
115
116 public Config() {
117 this(null);
118 }
119
120
121
122
123
124
125
126
127 public Config(Config defaultConfig) {
128 baseConfig = defaultConfig;
129 state = new AtomicReference<>(newState());
130 }
131
132
133
134
135
136
137
138
139
140 public static void setTypedConfigGetter(TypedConfigGetter getter) {
141 typedGetter = getter == null ? DEFAULT_GETTER : getter;
142 }
143
144
145
146
147
148
149
150
151 static String escapeValue(String x) {
152 if (x.isEmpty()) {
153 return "";
154 }
155
156 boolean needQuote = x.charAt(0) == ' ' || x.charAt(x.length() - 1) == ' ';
157 StringBuilder r = new StringBuilder(x.length());
158 for (int k = 0; k < x.length(); k++) {
159 char c = x.charAt(k);
160
161
162
163
164
165
166
167
168
169
170
171
172 switch (c) {
173 case '\0':
174
175
176
177 throw new IllegalArgumentException(
178 JGitText.get().configValueContainsNullByte);
179
180 case '\n':
181 r.append('\\').append('n');
182 break;
183
184 case '\t':
185 r.append('\\').append('t');
186 break;
187
188 case '\b':
189
190
191
192 r.append('\\').append('b');
193 break;
194
195 case '\\':
196 r.append('\\').append('\\');
197 break;
198
199 case '"':
200 r.append('\\').append('"');
201 break;
202
203 case '#':
204 case ';':
205 needQuote = true;
206 r.append(c);
207 break;
208
209 default:
210 r.append(c);
211 break;
212 }
213 }
214
215 return needQuote ? '"' + r.toString() + '"' : r.toString();
216 }
217
218 static String escapeSubsection(String x) {
219 if (x.isEmpty()) {
220 return "\"\"";
221 }
222
223 StringBuilder r = new StringBuilder(x.length() + 2).append('"');
224 for (int k = 0; k < x.length(); k++) {
225 char c = x.charAt(k);
226
227
228
229 switch (c) {
230 case '\0':
231 throw new IllegalArgumentException(
232 JGitText.get().configSubsectionContainsNullByte);
233
234 case '\n':
235 throw new IllegalArgumentException(
236 JGitText.get().configSubsectionContainsNewline);
237
238 case '\\':
239 case '"':
240 r.append('\\').append(c);
241 break;
242
243 default:
244 r.append(c);
245 break;
246 }
247 }
248
249 return r.append('"').toString();
250 }
251
252
253
254
255
256
257
258
259
260
261
262
263 public int getInt(final String section, final String name,
264 final int defaultValue) {
265 return typedGetter.getInt(this, section, null, name, defaultValue);
266 }
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281 public int getInt(final String section, String subsection,
282 final String name, final int defaultValue) {
283 return typedGetter.getInt(this, section, subsection, name,
284 defaultValue);
285 }
286
287
288
289
290
291
292
293
294
295
296
297
298 public long getLong(String section, String name, long defaultValue) {
299 return typedGetter.getLong(this, section, null, name, defaultValue);
300 }
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315 public long getLong(final String section, String subsection,
316 final String name, final long defaultValue) {
317 return typedGetter.getLong(this, section, subsection, name,
318 defaultValue);
319 }
320
321
322
323
324
325
326
327
328
329
330
331
332
333 public boolean getBoolean(final String section, final String name,
334 final boolean defaultValue) {
335 return typedGetter.getBoolean(this, section, null, name, defaultValue);
336 }
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352 public boolean getBoolean(final String section, String subsection,
353 final String name, final boolean defaultValue) {
354 return typedGetter.getBoolean(this, section, subsection, name,
355 defaultValue);
356 }
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371 public <T extends Enum<?>> T getEnum(final String section,
372 final String subsection, final String name, final T defaultValue) {
373 final T[] all = allValuesOf(defaultValue);
374 return typedGetter.getEnum(this, all, section, subsection, name,
375 defaultValue);
376 }
377
378 @SuppressWarnings("unchecked")
379 private static <T> T[] allValuesOf(T value) {
380 try {
381 return (T[]) value.getClass().getMethod("values").invoke(null);
382 } catch (Exception err) {
383 String typeName = value.getClass().getName();
384 String msg = MessageFormat.format(
385 JGitText.get().enumValuesNotAvailable, typeName);
386 throw new IllegalArgumentException(msg, err);
387 }
388 }
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406 public <T extends Enum<?>> T getEnum(final T[] all, final String section,
407 final String subsection, final String name, final T defaultValue) {
408 return typedGetter.getEnum(this, all, section, subsection, name,
409 defaultValue);
410 }
411
412
413
414
415
416
417
418
419
420
421
422
423 public String getString(final String section, String subsection,
424 final String name) {
425 return getRawString(section, subsection, name);
426 }
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442 public String[] getStringList(final String section, String subsection,
443 final String name) {
444 String[] base;
445 if (baseConfig != null)
446 base = baseConfig.getStringList(section, subsection, name);
447 else
448 base = EMPTY_STRING_ARRAY;
449
450 String[] self = getRawStringList(section, subsection, name);
451 if (self == null)
452 return base;
453 if (base.length == 0)
454 return self;
455 String[] res = new String[base.length + self.length];
456 int n = base.length;
457 System.arraycopy(base, 0, res, 0, n);
458 System.arraycopy(self, 0, res, n, self.length);
459 return res;
460 }
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481 public long getTimeUnit(String section, String subsection, String name,
482 long defaultValue, TimeUnit wantUnit) {
483 return typedGetter.getTimeUnit(this, section, subsection, name,
484 defaultValue, wantUnit);
485 }
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501 public List<RefSpec> getRefSpecs(String section, String subsection,
502 String name) {
503 return typedGetter.getRefSpecs(this, section, subsection, name);
504 }
505
506
507
508
509
510
511
512
513
514
515
516
517
518 public Set<String> getSubsections(String section) {
519 return getState().getSubsections(section);
520 }
521
522
523
524
525
526
527
528
529
530 public Set<String> getSections() {
531 return getState().getSections();
532 }
533
534
535
536
537
538
539
540
541 public Set<String> getNames(String section) {
542 return getNames(section, null);
543 }
544
545
546
547
548
549
550
551
552
553
554 public Set<String> getNames(String section, String subsection) {
555 return getState().getNames(section, subsection);
556 }
557
558
559
560
561
562
563
564
565
566
567
568
569 public Set<String> getNames(String section, boolean recursive) {
570 return getState().getNames(section, null, recursive);
571 }
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586 public Set<String> getNames(String section, String subsection,
587 boolean recursive) {
588 return getState().getNames(section, subsection, recursive);
589 }
590
591
592
593
594
595
596
597
598
599
600
601
602
603 @SuppressWarnings("unchecked")
604 public <T> T get(SectionParser<T> parser) {
605 final ConfigSnapshot myState = getState();
606 T obj = (T) myState.cache.get(parser);
607 if (obj == null) {
608 obj = parser.parse(this);
609 myState.cache.put(parser, obj);
610 }
611 return obj;
612 }
613
614
615
616
617
618
619
620
621
622
623
624 public void uncache(SectionParser<?> parser) {
625 state.get().cache.remove(parser);
626 }
627
628
629
630
631
632
633
634
635
636
637
638
639 public ListenerHandle addChangeListener(ConfigChangedListener listener) {
640 return listeners.addConfigChangedListener(listener);
641 }
642
643
644
645
646
647
648
649
650
651
652
653
654
655 protected boolean notifyUponTransientChanges() {
656 return true;
657 }
658
659
660
661
662 protected void fireConfigChangedEvent() {
663 listeners.dispatch(new ConfigChangedEvent());
664 }
665
666 String getRawString(final String section, final String subsection,
667 final String name) {
668 String[] lst = getRawStringList(section, subsection, name);
669 if (lst != null) {
670 return lst[lst.length - 1];
671 } else if (baseConfig != null) {
672 return baseConfig.getRawString(section, subsection, name);
673 } else {
674 return null;
675 }
676 }
677
678 private String[] getRawStringList(String section, String subsection,
679 String name) {
680 return state.get().get(section, subsection, name);
681 }
682
683 private ConfigSnapshot getState() {
684 ConfigSnapshot cur, upd;
685 do {
686 cur = state.get();
687 final ConfigSnapshot base = getBaseState();
688 if (cur.baseState == base)
689 return cur;
690 upd = new ConfigSnapshot(cur.entryList, base);
691 } while (!state.compareAndSet(cur, upd));
692 return upd;
693 }
694
695 private ConfigSnapshot getBaseState() {
696 return baseConfig != null ? baseConfig.getState() : null;
697 }
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717 public void setInt(final String section, final String subsection,
718 final String name, final int value) {
719 setLong(section, subsection, name, value);
720 }
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740 public void setLong(final String section, final String subsection,
741 final String name, final long value) {
742 final String s;
743
744 if (value >= GiB && (value % GiB) == 0)
745 s = String.valueOf(value / GiB) + "g";
746 else if (value >= MiB && (value % MiB) == 0)
747 s = String.valueOf(value / MiB) + "m";
748 else if (value >= KiB && (value % KiB) == 0)
749 s = String.valueOf(value / KiB) + "k";
750 else
751 s = String.valueOf(value);
752
753 setString(section, subsection, name, s);
754 }
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774 public void setBoolean(final String section, final String subsection,
775 final String name, final boolean value) {
776 setString(section, subsection, name, value ? "true" : "false");
777 }
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797 public <T extends Enum<?>> void setEnum(final String section,
798 final String subsection, final String name, final T value) {
799 String n;
800 if (value instanceof ConfigEnum)
801 n = ((ConfigEnum) value).toConfigValue();
802 else
803 n = value.name().toLowerCase(Locale.ROOT).replace('_', ' ');
804 setString(section, subsection, name, n);
805 }
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825 public void setString(final String section, final String subsection,
826 final String name, final String value) {
827 setStringList(section, subsection, name, Collections
828 .singletonList(value));
829 }
830
831
832
833
834
835
836
837
838
839
840
841 public void unset(final String section, final String subsection,
842 final String name) {
843 setStringList(section, subsection, name, Collections
844 .<String> emptyList());
845 }
846
847
848
849
850
851
852
853
854
855 public void unsetSection(String section, String subsection) {
856 ConfigSnapshot src, res;
857 do {
858 src = state.get();
859 res = unsetSection(src, section, subsection);
860 } while (!state.compareAndSet(src, res));
861 }
862
863 private ConfigSnapshot unsetSection(final ConfigSnapshot srcState,
864 final String section,
865 final String subsection) {
866 final int max = srcState.entryList.size();
867 final ArrayList<ConfigLine> r = new ArrayList<>(max);
868
869 boolean lastWasMatch = false;
870 for (ConfigLine e : srcState.entryList) {
871 if (e.match(section, subsection)) {
872
873 lastWasMatch = true;
874 continue;
875 }
876
877 if (lastWasMatch && e.section == null && e.subsection == null)
878 continue;
879 r.add(e);
880 }
881
882 return newState(r);
883 }
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903 public void setStringList(final String section, final String subsection,
904 final String name, final List<String> values) {
905 ConfigSnapshot src, res;
906 do {
907 src = state.get();
908 res = replaceStringList(src, section, subsection, name, values);
909 } while (!state.compareAndSet(src, res));
910 if (notifyUponTransientChanges())
911 fireConfigChangedEvent();
912 }
913
914 private ConfigSnapshot replaceStringList(final ConfigSnapshot srcState,
915 final String section, final String subsection, final String name,
916 final List<String> values) {
917 final List<ConfigLine> entries = copy(srcState, values);
918 int entryIndex = 0;
919 int valueIndex = 0;
920 int insertPosition = -1;
921
922
923
924 while (entryIndex < entries.size() && valueIndex < values.size()) {
925 final ConfigLine e = entries.get(entryIndex);
926 if (e.match(section, subsection, name)) {
927 entries.set(entryIndex, e.forValue(values.get(valueIndex++)));
928 insertPosition = entryIndex + 1;
929 }
930 entryIndex++;
931 }
932
933
934
935 if (valueIndex == values.size() && entryIndex < entries.size()) {
936 while (entryIndex < entries.size()) {
937 final ConfigLine e = entries.get(entryIndex++);
938 if (e.match(section, subsection, name))
939 entries.remove(--entryIndex);
940 }
941 }
942
943
944
945 if (valueIndex < values.size() && entryIndex == entries.size()) {
946 if (insertPosition < 0) {
947
948
949
950
951 insertPosition = findSectionEnd(entries, section, subsection);
952 }
953 if (insertPosition < 0) {
954
955
956
957 final ConfigLine e = new ConfigLine();
958 e.section = section;
959 e.subsection = subsection;
960 entries.add(e);
961 insertPosition = entries.size();
962 }
963 while (valueIndex < values.size()) {
964 final ConfigLine e = new ConfigLine();
965 e.section = section;
966 e.subsection = subsection;
967 e.name = name;
968 e.value = values.get(valueIndex++);
969 entries.add(insertPosition++, e);
970 }
971 }
972
973 return newState(entries);
974 }
975
976 private static List<ConfigLine> copy(final ConfigSnapshot src,
977 final List<String> values) {
978
979
980
981 final int max = src.entryList.size() + values.size() + 1;
982 final ArrayList<ConfigLine> r = new ArrayList<>(max);
983 r.addAll(src.entryList);
984 return r;
985 }
986
987 private static int findSectionEnd(final List<ConfigLine> entries,
988 final String section, final String subsection) {
989 for (int i = 0; i < entries.size(); i++) {
990 ConfigLine e = entries.get(i);
991 if (e.match(section, subsection, null)) {
992 i++;
993 while (i < entries.size()) {
994 e = entries.get(i);
995 if (e.match(section, subsection, e.name))
996 i++;
997 else
998 break;
999 }
1000 return i;
1001 }
1002 }
1003 return -1;
1004 }
1005
1006
1007
1008
1009
1010
1011 public String toText() {
1012 final StringBuilder out = new StringBuilder();
1013 for (ConfigLine e : state.get().entryList) {
1014 if (e.prefix != null)
1015 out.append(e.prefix);
1016 if (e.section != null && e.name == null) {
1017 out.append('[');
1018 out.append(e.section);
1019 if (e.subsection != null) {
1020 out.append(' ');
1021 String escaped = escapeValue(e.subsection);
1022
1023 boolean quoted = escaped.startsWith("\"")
1024 && escaped.endsWith("\"");
1025 if (!quoted)
1026 out.append('"');
1027 out.append(escaped);
1028 if (!quoted)
1029 out.append('"');
1030 }
1031 out.append(']');
1032 } else if (e.section != null && e.name != null) {
1033 if (e.prefix == null || "".equals(e.prefix))
1034 out.append('\t');
1035 out.append(e.name);
1036 if (MAGIC_EMPTY_VALUE != e.value) {
1037 out.append(" =");
1038 if (e.value != null) {
1039 out.append(' ');
1040 out.append(escapeValue(e.value));
1041 }
1042 }
1043 if (e.suffix != null)
1044 out.append(' ');
1045 }
1046 if (e.suffix != null)
1047 out.append(e.suffix);
1048 out.append('\n');
1049 }
1050 return out.toString();
1051 }
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062 public void fromText(String text) throws ConfigInvalidException {
1063 state.set(newState(fromTextRecurse(text, 1)));
1064 }
1065
1066 private List<ConfigLine> fromTextRecurse(String text, int depth)
1067 throws ConfigInvalidException {
1068 if (depth > MAX_DEPTH) {
1069 throw new ConfigInvalidException(
1070 JGitText.get().tooManyIncludeRecursions);
1071 }
1072 final List<ConfigLine> newEntries = new ArrayList<>();
1073 final StringReader in = new StringReader(text);
1074 ConfigLine last = null;
1075 ConfigLine e = new ConfigLine();
1076 for (;;) {
1077 int input = in.read();
1078 if (-1 == input) {
1079 if (e.section != null)
1080 newEntries.add(e);
1081 break;
1082 }
1083
1084 final char c = (char) input;
1085 if ('\n' == c) {
1086
1087 newEntries.add(e);
1088 if (e.section != null)
1089 last = e;
1090 e = new ConfigLine();
1091
1092 } else if (e.suffix != null) {
1093
1094 e.suffix += c;
1095
1096 } else if (';' == c || '#' == c) {
1097
1098 e.suffix = String.valueOf(c);
1099
1100 } else if (e.section == null && Character.isWhitespace(c)) {
1101
1102 if (e.prefix == null)
1103 e.prefix = "";
1104 e.prefix += c;
1105
1106 } else if ('[' == c) {
1107
1108 e.section = readSectionName(in);
1109 input = in.read();
1110 if ('"' == input) {
1111 e.subsection = readSubsectionName(in);
1112 input = in.read();
1113 }
1114 if (']' != input)
1115 throw new ConfigInvalidException(JGitText.get().badGroupHeader);
1116 e.suffix = "";
1117
1118 } else if (last != null) {
1119
1120 e.section = last.section;
1121 e.subsection = last.subsection;
1122 in.reset();
1123 e.name = readKeyName(in);
1124 if (e.name.endsWith("\n")) {
1125 e.name = e.name.substring(0, e.name.length() - 1);
1126 e.value = MAGIC_EMPTY_VALUE;
1127 } else
1128 e.value = readValue(in);
1129
1130 if (e.section.equalsIgnoreCase("include")) {
1131 addIncludedConfig(newEntries, e, depth);
1132 }
1133 } else
1134 throw new ConfigInvalidException(JGitText.get().invalidLineInConfigFile);
1135 }
1136
1137 return newEntries;
1138 }
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151 @Nullable
1152 protected byte[] readIncludedConfig(String relPath)
1153 throws ConfigInvalidException {
1154 return null;
1155 }
1156
1157 private void addIncludedConfig(final List<ConfigLine> newEntries,
1158 ConfigLine line, int depth) throws ConfigInvalidException {
1159 if (!line.name.equalsIgnoreCase("path") ||
1160 line.value == null || line.value.equals(MAGIC_EMPTY_VALUE)) {
1161 throw new ConfigInvalidException(MessageFormat.format(
1162 JGitText.get().invalidLineInConfigFileWithParam, line));
1163 }
1164 byte[] bytes = readIncludedConfig(line.value);
1165 if (bytes == null) {
1166 return;
1167 }
1168
1169 String decoded;
1170 if (isUtf8(bytes)) {
1171 decoded = RawParseUtils.decode(UTF_8, bytes, 3, bytes.length);
1172 } else {
1173 decoded = RawParseUtils.decode(bytes);
1174 }
1175 try {
1176 newEntries.addAll(fromTextRecurse(decoded, depth + 1));
1177 } catch (ConfigInvalidException e) {
1178 throw new ConfigInvalidException(MessageFormat
1179 .format(JGitText.get().cannotReadFile, line.value), e);
1180 }
1181 }
1182
1183 private ConfigSnapshot newState() {
1184 return new ConfigSnapshot(Collections.<ConfigLine> emptyList(),
1185 getBaseState());
1186 }
1187
1188 private ConfigSnapshot newState(List<ConfigLine> entries) {
1189 return new ConfigSnapshot(Collections.unmodifiableList(entries),
1190 getBaseState());
1191 }
1192
1193
1194
1195
1196 protected void clear() {
1197 state.set(newState());
1198 }
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208 protected boolean isUtf8(final byte[] bytes) {
1209 return bytes.length >= 3 && bytes[0] == (byte) 0xEF
1210 && bytes[1] == (byte) 0xBB && bytes[2] == (byte) 0xBF;
1211 }
1212
1213 private static String readSectionName(StringReader in)
1214 throws ConfigInvalidException {
1215 final StringBuilder name = new StringBuilder();
1216 for (;;) {
1217 int c = in.read();
1218 if (c < 0)
1219 throw new ConfigInvalidException(JGitText.get().unexpectedEndOfConfigFile);
1220
1221 if (']' == c) {
1222 in.reset();
1223 break;
1224 }
1225
1226 if (' ' == c || '\t' == c) {
1227 for (;;) {
1228 c = in.read();
1229 if (c < 0)
1230 throw new ConfigInvalidException(JGitText.get().unexpectedEndOfConfigFile);
1231
1232 if ('"' == c) {
1233 in.reset();
1234 break;
1235 }
1236
1237 if (' ' == c || '\t' == c)
1238 continue;
1239 throw new ConfigInvalidException(MessageFormat.format(JGitText.get().badSectionEntry, name));
1240 }
1241 break;
1242 }
1243
1244 if (Character.isLetterOrDigit((char) c) || '.' == c || '-' == c)
1245 name.append((char) c);
1246 else
1247 throw new ConfigInvalidException(MessageFormat.format(JGitText.get().badSectionEntry, name));
1248 }
1249 return name.toString();
1250 }
1251
1252 private static String readKeyName(StringReader in)
1253 throws ConfigInvalidException {
1254 final StringBuilder name = new StringBuilder();
1255 for (;;) {
1256 int c = in.read();
1257 if (c < 0)
1258 throw new ConfigInvalidException(JGitText.get().unexpectedEndOfConfigFile);
1259
1260 if ('=' == c)
1261 break;
1262
1263 if (' ' == c || '\t' == c) {
1264 for (;;) {
1265 c = in.read();
1266 if (c < 0)
1267 throw new ConfigInvalidException(JGitText.get().unexpectedEndOfConfigFile);
1268
1269 if ('=' == c)
1270 break;
1271
1272 if (';' == c || '#' == c || '\n' == c) {
1273 in.reset();
1274 break;
1275 }
1276
1277 if (' ' == c || '\t' == c)
1278 continue;
1279 throw new ConfigInvalidException(JGitText.get().badEntryDelimiter);
1280 }
1281 break;
1282 }
1283
1284 if (Character.isLetterOrDigit((char) c) || c == '-') {
1285
1286
1287
1288 name.append((char) c);
1289 } else if ('\n' == c) {
1290 in.reset();
1291 name.append((char) c);
1292 break;
1293 } else
1294 throw new ConfigInvalidException(MessageFormat.format(JGitText.get().badEntryName, name));
1295 }
1296 return name.toString();
1297 }
1298
1299 private static String readSubsectionName(StringReader in)
1300 throws ConfigInvalidException {
1301 StringBuilder r = new StringBuilder();
1302 for (;;) {
1303 int c = in.read();
1304 if (c < 0) {
1305 break;
1306 }
1307
1308 if ('\n' == c) {
1309 throw new ConfigInvalidException(
1310 JGitText.get().newlineInQuotesNotAllowed);
1311 }
1312 if ('\\' == c) {
1313 c = in.read();
1314 switch (c) {
1315 case -1:
1316 throw new ConfigInvalidException(JGitText.get().endOfFileInEscape);
1317
1318 case '\\':
1319 case '"':
1320 r.append((char) c);
1321 continue;
1322
1323 default:
1324
1325
1326 r.append((char) c);
1327 continue;
1328 }
1329 }
1330 if ('"' == c) {
1331 break;
1332 }
1333
1334 r.append((char) c);
1335 }
1336 return r.toString();
1337 }
1338
1339 private static String readValue(StringReader in)
1340 throws ConfigInvalidException {
1341 StringBuilder value = new StringBuilder();
1342 StringBuilder trailingSpaces = null;
1343 boolean quote = false;
1344 boolean inLeadingSpace = true;
1345
1346 for (;;) {
1347 int c = in.read();
1348 if (c < 0) {
1349 break;
1350 }
1351 if ('\n' == c) {
1352 if (quote) {
1353 throw new ConfigInvalidException(
1354 JGitText.get().newlineInQuotesNotAllowed);
1355 }
1356 in.reset();
1357 break;
1358 }
1359
1360 if (!quote && (';' == c || '#' == c)) {
1361 if (trailingSpaces != null) {
1362 trailingSpaces.setLength(0);
1363 }
1364 in.reset();
1365 break;
1366 }
1367
1368 char cc = (char) c;
1369 if (Character.isWhitespace(cc)) {
1370 if (inLeadingSpace) {
1371 continue;
1372 }
1373 if (trailingSpaces == null) {
1374 trailingSpaces = new StringBuilder();
1375 }
1376 trailingSpaces.append(cc);
1377 continue;
1378 } else {
1379 inLeadingSpace = false;
1380 if (trailingSpaces != null) {
1381 value.append(trailingSpaces);
1382 trailingSpaces.setLength(0);
1383 }
1384 }
1385
1386 if ('\\' == c) {
1387 c = in.read();
1388 switch (c) {
1389 case -1:
1390 throw new ConfigInvalidException(JGitText.get().endOfFileInEscape);
1391 case '\n':
1392 continue;
1393 case 't':
1394 value.append('\t');
1395 continue;
1396 case 'b':
1397 value.append('\b');
1398 continue;
1399 case 'n':
1400 value.append('\n');
1401 continue;
1402 case '\\':
1403 value.append('\\');
1404 continue;
1405 case '"':
1406 value.append('"');
1407 continue;
1408 default:
1409 throw new ConfigInvalidException(MessageFormat.format(
1410 JGitText.get().badEscape,
1411 Character.valueOf(((char) c))));
1412 }
1413 }
1414
1415 if ('"' == c) {
1416 quote = !quote;
1417 continue;
1418 }
1419
1420 value.append(cc);
1421 }
1422 return value.length() > 0 ? value.toString() : null;
1423 }
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439 public static interface SectionParser<T> {
1440
1441
1442
1443
1444
1445
1446
1447 T parse(Config cfg);
1448 }
1449
1450 private static class StringReader {
1451 private final char[] buf;
1452
1453 private int pos;
1454
1455 StringReader(String in) {
1456 buf = in.toCharArray();
1457 }
1458
1459 int read() {
1460 try {
1461 return buf[pos++];
1462 } catch (ArrayIndexOutOfBoundsException e) {
1463 pos = buf.length;
1464 return -1;
1465 }
1466 }
1467
1468 void reset() {
1469 pos--;
1470 }
1471 }
1472
1473
1474
1475
1476
1477
1478 public static interface ConfigEnum {
1479
1480
1481
1482
1483
1484 String toConfigValue();
1485
1486
1487
1488
1489
1490
1491
1492
1493 boolean matchConfigValue(String in);
1494 }
1495 }