Home | History | Annotate | Download | only in internal
      1 /*
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.inputmethod.keyboard.internal;
     18 
     19 import static com.android.inputmethod.keyboard.internal.KeyboardIconsSet.ICON_UNDEFINED;
     20 import static com.android.inputmethod.latin.Constants.CODE_OUTPUT_TEXT;
     21 import static com.android.inputmethod.latin.Constants.CODE_UNSPECIFIED;
     22 
     23 import android.content.Context;
     24 import android.content.res.Resources;
     25 import android.test.AndroidTestCase;
     26 import android.test.suitebuilder.annotation.SmallTest;
     27 
     28 import com.android.inputmethod.latin.Constants;
     29 import com.android.inputmethod.latin.utils.RunInLocale;
     30 
     31 import java.util.Arrays;
     32 import java.util.Locale;
     33 
     34 @SmallTest
     35 public class KeySpecParserTests extends AndroidTestCase {
     36     private final static Locale TEST_LOCALE = Locale.ENGLISH;
     37     final KeyboardCodesSet mCodesSet = new KeyboardCodesSet();
     38     final KeyboardTextsSet mTextsSet = new KeyboardTextsSet();
     39 
     40     private static final String CODE_SETTINGS = "!code/key_settings";
     41     private static final String ICON_SETTINGS = "!icon/settings_key";
     42     private static final String CODE_SETTINGS_UPPERCASE = CODE_SETTINGS.toUpperCase(Locale.ROOT);
     43     private static final String ICON_SETTINGS_UPPERCASE = ICON_SETTINGS.toUpperCase(Locale.ROOT);
     44     private static final String CODE_NON_EXISTING = "!code/non_existing";
     45     private static final String ICON_NON_EXISTING = "!icon/non_existing";
     46 
     47     private int mCodeSettings;
     48     private int mCodeActionNext;
     49     private int mSettingsIconId;
     50 
     51     @Override
     52     protected void setUp() throws Exception {
     53         super.setUp();
     54 
     55         final String language = TEST_LOCALE.getLanguage();
     56         mCodesSet.setLanguage(language);
     57         mTextsSet.setLanguage(language);
     58         final Context context = getContext();
     59         new RunInLocale<Void>() {
     60             @Override
     61             protected Void job(final Resources res) {
     62                 mTextsSet.loadStringResources(context);
     63                 return null;
     64             }
     65         }.runInLocale(context.getResources(), TEST_LOCALE);
     66 
     67         mCodeSettings = KeySpecParser.parseCode(
     68                 CODE_SETTINGS, mCodesSet, CODE_UNSPECIFIED);
     69         mCodeActionNext = KeySpecParser.parseCode(
     70                 "!code/key_action_next", mCodesSet, CODE_UNSPECIFIED);
     71         mSettingsIconId = KeySpecParser.getIconId(ICON_SETTINGS);
     72     }
     73 
     74     private void assertParser(String message, String moreKeySpec, String expectedLabel,
     75             String expectedOutputText, int expectedIcon, int expectedCode) {
     76         final String labelResolved = KeySpecParser.resolveTextReference(moreKeySpec, mTextsSet);
     77         final MoreKeySpec spec = new MoreKeySpec(labelResolved, false /* needsToUpperCase */,
     78                 Locale.US, mCodesSet);
     79         assertEquals(message + " [label]", expectedLabel, spec.mLabel);
     80         assertEquals(message + " [ouptputText]", expectedOutputText, spec.mOutputText);
     81         assertEquals(message + " [icon]",
     82                 KeyboardIconsSet.getIconName(expectedIcon),
     83                 KeyboardIconsSet.getIconName(spec.mIconId));
     84         assertEquals(message + " [code]",
     85                 Constants.printableCode(expectedCode),
     86                 Constants.printableCode(spec.mCode));
     87     }
     88 
     89     private void assertParserError(String message, String moreKeySpec, String expectedLabel,
     90             String expectedOutputText, int expectedIcon, int expectedCode) {
     91         try {
     92             assertParser(message, moreKeySpec, expectedLabel, expectedOutputText, expectedIcon,
     93                     expectedCode);
     94             fail(message);
     95         } catch (Exception pcpe) {
     96             // success.
     97         }
     98     }
     99 
    100     // \U001d11e: MUSICAL SYMBOL G CLEF
    101     private static final String PAIR1 = "\ud834\udd1e";
    102     private static final int CODE1 = PAIR1.codePointAt(0);
    103     // \U001d122: MUSICAL SYMBOL F CLEF
    104     private static final String PAIR2 = "\ud834\udd22";
    105     private static final int CODE2 = PAIR2.codePointAt(0);
    106     // \U002f8a6: CJK COMPATIBILITY IDEOGRAPH-2F8A6; variant character of \u6148.
    107     private static final String PAIR3 = "\ud87e\udca6";
    108     private static final String SURROGATE1 = PAIR1 + PAIR2;
    109     private static final String SURROGATE2 = PAIR1 + PAIR2 + PAIR3;
    110 
    111     public void testSingleLetter() {
    112         assertParser("Single letter", "a",
    113                 "a", null, ICON_UNDEFINED, 'a');
    114         assertParser("Single surrogate", PAIR1,
    115                 PAIR1, null, ICON_UNDEFINED, CODE1);
    116         assertParser("Single escaped bar", "\\|",
    117                 "|", null, ICON_UNDEFINED, '|');
    118         assertParser("Single escaped escape", "\\\\",
    119                 "\\", null, ICON_UNDEFINED, '\\');
    120         assertParser("Single comma", ",",
    121                 ",", null, ICON_UNDEFINED, ',');
    122         assertParser("Single escaped comma", "\\,",
    123                 ",", null, ICON_UNDEFINED, ',');
    124         assertParser("Single escaped letter", "\\a",
    125                 "a", null, ICON_UNDEFINED, 'a');
    126         assertParser("Single escaped surrogate", "\\" + PAIR2,
    127                 PAIR2, null, ICON_UNDEFINED, CODE2);
    128         assertParser("Single bang", "!",
    129                 "!", null, ICON_UNDEFINED, '!');
    130         assertParser("Single escaped bang", "\\!",
    131                 "!", null, ICON_UNDEFINED, '!');
    132         assertParser("Single output text letter", "a|a",
    133                 "a", null, ICON_UNDEFINED, 'a');
    134         assertParser("Single surrogate pair outputText", "G Clef|" + PAIR1,
    135                 "G Clef", null, ICON_UNDEFINED, CODE1);
    136         assertParser("Single letter with outputText", "a|abc",
    137                 "a", "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    138         assertParser("Single letter with surrogate outputText", "a|" + SURROGATE1,
    139                 "a", SURROGATE1, ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    140         assertParser("Single surrogate with outputText", PAIR3 + "|abc",
    141                 PAIR3, "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    142         assertParser("Single letter with escaped outputText", "a|a\\|c",
    143                 "a", "a|c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    144         assertParser("Single letter with escaped surrogate outputText",
    145                 "a|" + PAIR1 + "\\|" + PAIR2,
    146                 "a", PAIR1 + "|" + PAIR2, ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    147         assertParser("Single letter with comma outputText", "a|a,b",
    148                 "a", "a,b", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    149         assertParser("Single letter with escaped comma outputText", "a|a\\,b",
    150                 "a", "a,b", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    151         assertParser("Single letter with outputText starts with bang", "a|!bc",
    152                 "a", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    153         assertParser("Single letter with surrogate outputText starts with bang", "a|!" + SURROGATE2,
    154                 "a", "!" + SURROGATE2, ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    155         assertParser("Single letter with outputText contains bang", "a|a!c",
    156                 "a", "a!c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    157         assertParser("Single letter with escaped bang outputText", "a|\\!bc",
    158                 "a", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    159         assertParser("Single escaped escape with single outputText", "\\\\|\\\\",
    160                 "\\", null, ICON_UNDEFINED, '\\');
    161         assertParser("Single escaped bar with single outputText", "\\||\\|",
    162                 "|", null, ICON_UNDEFINED, '|');
    163         assertParser("Single letter with code", "a|" + CODE_SETTINGS,
    164                 "a", null, ICON_UNDEFINED, mCodeSettings);
    165     }
    166 
    167     public void testLabel() {
    168         assertParser("Simple label", "abc",
    169                 "abc", "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    170         assertParser("Simple surrogate label", SURROGATE1,
    171                 SURROGATE1, SURROGATE1, ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    172         assertParser("Label with escaped bar", "a\\|c",
    173                 "a|c", "a|c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    174         assertParser("Surrogate label with escaped bar", PAIR1 + "\\|" + PAIR2,
    175                 PAIR1 + "|" + PAIR2, PAIR1 + "|" + PAIR2,
    176                 ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    177         assertParser("Label with escaped escape", "a\\\\c",
    178                 "a\\c", "a\\c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    179         assertParser("Label with comma", "a,c",
    180                 "a,c", "a,c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    181         assertParser("Label with escaped comma", "a\\,c",
    182                 "a,c", "a,c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    183         assertParser("Label starts with bang", "!bc",
    184                 "!bc", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    185         assertParser("Surrogate label starts with bang", "!" + SURROGATE1,
    186                 "!" + SURROGATE1, "!" + SURROGATE1, ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    187         assertParser("Label contains bang", "a!c",
    188                 "a!c", "a!c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    189         assertParser("Label with escaped bang", "\\!bc",
    190                 "!bc", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    191         assertParser("Label with escaped letter", "\\abc",
    192                 "abc", "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    193         assertParser("Label with outputText", "abc|def",
    194                 "abc", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    195         assertParser("Label with comma and outputText", "a,c|def",
    196                 "a,c", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    197         assertParser("Escaped comma label with outputText", "a\\,c|def",
    198                 "a,c", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    199         assertParser("Escaped label with outputText", "a\\|c|def",
    200                 "a|c", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    201         assertParser("Label with escaped bar outputText", "abc|d\\|f",
    202                 "abc", "d|f", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    203         assertParser("Escaped escape label with outputText", "a\\\\|def",
    204                 "a\\", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    205         assertParser("Label starts with bang and outputText", "!bc|def",
    206                 "!bc", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    207         assertParser("Label contains bang label and outputText", "a!c|def",
    208                 "a!c", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    209         assertParser("Escaped bang label with outputText", "\\!bc|def",
    210                 "!bc", "def", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    211         assertParser("Label with comma outputText", "abc|a,b",
    212                 "abc", "a,b", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    213         assertParser("Label with escaped comma outputText", "abc|a\\,b",
    214                 "abc", "a,b", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    215         assertParser("Label with outputText starts with bang", "abc|!bc",
    216                 "abc", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    217         assertParser("Label with outputText contains bang", "abc|a!c",
    218                 "abc", "a!c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    219         assertParser("Label with escaped bang outputText", "abc|\\!bc",
    220                 "abc", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    221         assertParser("Label with escaped bar outputText", "abc|d\\|f",
    222                 "abc", "d|f", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    223         assertParser("Escaped bar label with escaped bar outputText", "a\\|c|d\\|f",
    224                 "a|c", "d|f", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    225         assertParser("Label with code", "abc|" + CODE_SETTINGS,
    226                 "abc", null, ICON_UNDEFINED, mCodeSettings);
    227         assertParser("Escaped label with code", "a\\|c|" + CODE_SETTINGS,
    228                 "a|c", null, ICON_UNDEFINED, mCodeSettings);
    229     }
    230 
    231     public void testIconAndCode() {
    232         assertParser("Icon with outputText", ICON_SETTINGS + "|abc",
    233                 null, "abc", mSettingsIconId, CODE_OUTPUT_TEXT);
    234         assertParser("Icon with outputText starts with bang", ICON_SETTINGS + "|!bc",
    235                 null, "!bc", mSettingsIconId, CODE_OUTPUT_TEXT);
    236         assertParser("Icon with outputText contains bang", ICON_SETTINGS + "|a!c",
    237                 null, "a!c", mSettingsIconId, CODE_OUTPUT_TEXT);
    238         assertParser("Icon with escaped bang outputText", ICON_SETTINGS + "|\\!bc",
    239                 null, "!bc", mSettingsIconId, CODE_OUTPUT_TEXT);
    240         assertParser("Label starts with bang and code", "!bc|" + CODE_SETTINGS,
    241                 "!bc", null, ICON_UNDEFINED, mCodeSettings);
    242         assertParser("Label contains bang and code", "a!c|" + CODE_SETTINGS,
    243                 "a!c", null, ICON_UNDEFINED, mCodeSettings);
    244         assertParser("Escaped bang label with code", "\\!bc|" + CODE_SETTINGS,
    245                 "!bc", null, ICON_UNDEFINED, mCodeSettings);
    246         assertParser("Icon with code", ICON_SETTINGS + "|" + CODE_SETTINGS,
    247                 null, null, mSettingsIconId, mCodeSettings);
    248     }
    249 
    250     public void testResourceReference() {
    251         assertParser("Settings as more key", "!text/settings_as_more_key",
    252                 null, null, mSettingsIconId, mCodeSettings);
    253 
    254         assertParser("Action next as more key", "!text/label_next_key|!code/key_action_next",
    255                 "Next", null, ICON_UNDEFINED, mCodeActionNext);
    256 
    257         assertParser("Popular domain",
    258                 "!text/keylabel_for_popular_domain|!text/keylabel_for_popular_domain ",
    259                 ".com", ".com ", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    260     }
    261 
    262     public void testFormatError() {
    263         assertParserError("Empty spec", "", null,
    264                 null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    265         assertParserError("Empty label with outputText", "|a",
    266                 null, "a", ICON_UNDEFINED, CODE_UNSPECIFIED);
    267         assertParserError("Empty label with code", "|" + CODE_SETTINGS,
    268                 null, null, ICON_UNDEFINED, mCodeSettings);
    269         assertParserError("Empty outputText with label", "a|",
    270                 "a", null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    271         assertParserError("Empty outputText with icon", ICON_SETTINGS + "|",
    272                 null, null, mSettingsIconId, CODE_UNSPECIFIED);
    273         assertParserError("Empty icon and code", "|",
    274                 null, null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    275         assertParserError("Icon without code", ICON_SETTINGS,
    276                 null, null, mSettingsIconId, CODE_UNSPECIFIED);
    277         assertParserError("Non existing icon", ICON_NON_EXISTING + "|abc",
    278                 null, "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    279         assertParserError("Non existing code", "abc|" + CODE_NON_EXISTING,
    280                 "abc", null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    281         assertParserError("Third bar at end", "a|b|",
    282                 "a", null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    283         assertParserError("Multiple bar", "a|b|c",
    284                 "a", null, ICON_UNDEFINED, CODE_UNSPECIFIED);
    285         assertParserError("Multiple bar with label and code", "a|" + CODE_SETTINGS + "|c",
    286                 "a", null, ICON_UNDEFINED, mCodeSettings);
    287         assertParserError("Multiple bar with icon and outputText", ICON_SETTINGS + "|b|c",
    288                 null, null, mSettingsIconId, CODE_UNSPECIFIED);
    289         assertParserError("Multiple bar with icon and code",
    290                 ICON_SETTINGS + "|" + CODE_SETTINGS + "|c",
    291                 null, null, mSettingsIconId, mCodeSettings);
    292     }
    293 
    294     public void testUselessUpperCaseSpecifier() {
    295         assertParser("Single letter with CODE", "a|" + CODE_SETTINGS_UPPERCASE,
    296                 "a", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    297         assertParser("Label with CODE", "abc|" + CODE_SETTINGS_UPPERCASE,
    298                 "abc", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    299         assertParser("Escaped label with CODE", "a\\|c|" + CODE_SETTINGS_UPPERCASE,
    300                 "a|c", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    301         assertParser("ICON with outputText", ICON_SETTINGS_UPPERCASE + "|abc",
    302                 "!ICON/SETTINGS_KEY", "abc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    303         assertParser("ICON with outputText starts with bang", ICON_SETTINGS_UPPERCASE + "|!bc",
    304                 "!ICON/SETTINGS_KEY", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    305         assertParser("ICON with outputText contains bang", ICON_SETTINGS_UPPERCASE + "|a!c",
    306                 "!ICON/SETTINGS_KEY", "a!c", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    307         assertParser("ICON with escaped bang outputText", ICON_SETTINGS_UPPERCASE + "|\\!bc",
    308                 "!ICON/SETTINGS_KEY", "!bc", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    309         assertParser("Label starts with bang and CODE", "!bc|" + CODE_SETTINGS_UPPERCASE,
    310                 "!bc", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    311         assertParser("Label contains bang and CODE", "a!c|" + CODE_SETTINGS_UPPERCASE,
    312                 "a!c", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    313         assertParser("Escaped bang label with CODE", "\\!bc|" + CODE_SETTINGS_UPPERCASE,
    314                 "!bc", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    315         assertParser("ICON with CODE", ICON_SETTINGS_UPPERCASE + "|" + CODE_SETTINGS_UPPERCASE,
    316                 "!ICON/SETTINGS_KEY", "!CODE/KEY_SETTINGS", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    317         assertParser("SETTINGS AS MORE KEY", "!TEXT/SETTINGS_AS_MORE_KEY",
    318                 "!TEXT/SETTINGS_AS_MORE_KEY", "!TEXT/SETTINGS_AS_MORE_KEY", ICON_UNDEFINED,
    319                 CODE_OUTPUT_TEXT);
    320         assertParser("ACTION NEXT AS MORE KEY", "!TEXT/LABEL_NEXT_KEY|!CODE/KEY_ACTION_NEXT",
    321                 "!TEXT/LABEL_NEXT_KEY", "!CODE/KEY_ACTION_NEXT", ICON_UNDEFINED,
    322                 CODE_OUTPUT_TEXT);
    323         assertParser("POPULAR DOMAIN",
    324                 "!TEXT/KEYLABEL_FOR_POPULAR_DOMAIN|!TEXT/KEYLABEL_FOR_POPULAR_DOMAIN ",
    325                 "!TEXT/KEYLABEL_FOR_POPULAR_DOMAIN", "!TEXT/KEYLABEL_FOR_POPULAR_DOMAIN ",
    326                 ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    327         assertParserError("Empty label with CODE", "|" + CODE_SETTINGS_UPPERCASE,
    328                 null, null, ICON_UNDEFINED, mCodeSettings);
    329         assertParserError("Empty outputText with ICON", ICON_SETTINGS_UPPERCASE + "|",
    330                 null, null, mSettingsIconId, CODE_UNSPECIFIED);
    331         assertParser("ICON without code", ICON_SETTINGS_UPPERCASE,
    332                 "!ICON/SETTINGS_KEY", "!ICON/SETTINGS_KEY", ICON_UNDEFINED, CODE_OUTPUT_TEXT);
    333         assertParserError("Multiple bar with label and CODE", "a|" + CODE_SETTINGS_UPPERCASE + "|c",
    334                 "a", null, ICON_UNDEFINED, mCodeSettings);
    335         assertParserError("Multiple bar with ICON and outputText", ICON_SETTINGS_UPPERCASE + "|b|c",
    336                 null, null, mSettingsIconId, CODE_UNSPECIFIED);
    337         assertParserError("Multiple bar with ICON and CODE",
    338                 ICON_SETTINGS_UPPERCASE + "|" + CODE_SETTINGS_UPPERCASE + "|c",
    339                 null, null, mSettingsIconId, mCodeSettings);
    340     }
    341 
    342     private static void assertArrayEquals(String message, Object[] expected, Object[] actual) {
    343         if (expected == actual) {
    344             return;
    345         }
    346         if (expected == null || actual == null) {
    347             assertEquals(message, Arrays.toString(expected), Arrays.toString(actual));
    348             return;
    349         }
    350         if (expected.length != actual.length) {
    351             assertEquals(message + " [length]", Arrays.toString(expected), Arrays.toString(actual));
    352             return;
    353         }
    354         for (int i = 0; i < expected.length; i++) {
    355             assertEquals(message + " [" + i + "]",
    356                     Arrays.toString(expected), Arrays.toString(actual));
    357         }
    358     }
    359 
    360     private static void assertInsertAdditionalMoreKeys(String message, String[] moreKeys,
    361             String[] additionalMoreKeys, String[] expected) {
    362         final String[] actual =
    363                 KeySpecParser.insertAdditionalMoreKeys( moreKeys, additionalMoreKeys);
    364         assertArrayEquals(message, expected, actual);
    365     }
    366 
    367     public void testEmptyEntry() {
    368         assertInsertAdditionalMoreKeys("null more keys and null additons",
    369                 null,
    370                 null,
    371                 null);
    372         assertInsertAdditionalMoreKeys("null more keys and empty additons",
    373                 null,
    374                 new String[0],
    375                 null);
    376         assertInsertAdditionalMoreKeys("empty more keys and null additons",
    377                 new String[0],
    378                 null,
    379                 null);
    380         assertInsertAdditionalMoreKeys("empty more keys and empty additons",
    381                 new String[0],
    382                 new String[0],
    383                 null);
    384 
    385         assertInsertAdditionalMoreKeys("filter out empty more keys",
    386                 new String[] { null, "a", "", "b", null },
    387                 null,
    388                 new String[] { "a", "b" });
    389         assertInsertAdditionalMoreKeys("filter out empty additons",
    390                 new String[] { "a", "%", "b", "%", "c", "%", "d" },
    391                 new String[] { null, "A", "", "B", null },
    392                 new String[] { "a", "A", "b", "B", "c", "d" });
    393     }
    394 
    395     public void testInsertAdditionalMoreKeys() {
    396         // Escaped marker.
    397         assertInsertAdditionalMoreKeys("escaped marker",
    398                 new String[] { "\\%", "%-)" },
    399                 new String[] { "1", "2" },
    400                 new String[] { "1", "2", "\\%", "%-)" });
    401 
    402         // 0 more key.
    403         assertInsertAdditionalMoreKeys("null & null", null, null, null);
    404         assertInsertAdditionalMoreKeys("null & 1 additon",
    405                 null,
    406                 new String[] { "1" },
    407                 new String[] { "1" });
    408         assertInsertAdditionalMoreKeys("null & 2 additons",
    409                 null,
    410                 new String[] { "1", "2" },
    411                 new String[] { "1", "2" });
    412 
    413         // 0 additional more key.
    414         assertInsertAdditionalMoreKeys("1 more key & null",
    415                 new String[] { "A" },
    416                 null,
    417                 new String[] { "A" });
    418         assertInsertAdditionalMoreKeys("2 more keys & null",
    419                 new String[] { "A", "B" },
    420                 null,
    421                 new String[] { "A", "B" });
    422 
    423         // No marker.
    424         assertInsertAdditionalMoreKeys("1 more key & 1 addtional & no marker",
    425                 new String[] { "A" },
    426                 new String[] { "1" },
    427                 new String[] { "1", "A" });
    428         assertInsertAdditionalMoreKeys("1 more key & 2 addtionals & no marker",
    429                 new String[] { "A" },
    430                 new String[] { "1", "2" },
    431                 new String[] { "1", "2", "A" });
    432         assertInsertAdditionalMoreKeys("2 more keys & 1 addtional & no marker",
    433                 new String[] { "A", "B" },
    434                 new String[] { "1" },
    435                 new String[] { "1", "A", "B" });
    436         assertInsertAdditionalMoreKeys("2 more keys & 2 addtionals & no marker",
    437                 new String[] { "A", "B" },
    438                 new String[] { "1", "2" },
    439                 new String[] { "1", "2", "A", "B" });
    440 
    441         // 1 marker.
    442         assertInsertAdditionalMoreKeys("1 more key & 1 additon & marker at head",
    443                 new String[] { "%", "A" },
    444                 new String[] { "1" },
    445                 new String[] { "1", "A" });
    446         assertInsertAdditionalMoreKeys("1 more key & 1 additon & marker at tail",
    447                 new String[] { "A", "%" },
    448                 new String[] { "1" },
    449                 new String[] { "A", "1" });
    450         assertInsertAdditionalMoreKeys("2 more keys & 1 additon & marker at middle",
    451                 new String[] { "A", "%", "B" },
    452                 new String[] { "1" },
    453                 new String[] { "A", "1", "B" });
    454 
    455         // 1 marker & excess additional more keys.
    456         assertInsertAdditionalMoreKeys("1 more key & 2 additons & marker at head",
    457                 new String[] { "%", "A", "B" },
    458                 new String[] { "1", "2" },
    459                 new String[] { "1", "A", "B", "2" });
    460         assertInsertAdditionalMoreKeys("1 more key & 2 additons & marker at tail",
    461                 new String[] { "A", "B", "%" },
    462                 new String[] { "1", "2" },
    463                 new String[] { "A", "B", "1", "2" });
    464         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & marker at middle",
    465                 new String[] { "A", "%", "B" },
    466                 new String[] { "1", "2" },
    467                 new String[] { "A", "1", "B", "2" });
    468 
    469         // 2 markers.
    470         assertInsertAdditionalMoreKeys("0 more key & 2 addtional & 2 markers",
    471                 new String[] { "%", "%" },
    472                 new String[] { "1", "2" },
    473                 new String[] { "1", "2" });
    474         assertInsertAdditionalMoreKeys("1 more key & 2 addtional & 2 markers at head",
    475                 new String[] { "%", "%", "A" },
    476                 new String[] { "1", "2" },
    477                 new String[] { "1", "2", "A" });
    478         assertInsertAdditionalMoreKeys("1 more key & 2 addtional & 2 markers at tail",
    479                 new String[] { "A", "%", "%" },
    480                 new String[] { "1", "2" },
    481                 new String[] { "A", "1", "2" });
    482         assertInsertAdditionalMoreKeys("2 more keys & 2 addtional & 2 markers at middle",
    483                 new String[] { "A", "%", "%", "B" },
    484                 new String[] { "1", "2" },
    485                 new String[] { "A", "1", "2", "B" });
    486         assertInsertAdditionalMoreKeys("2 more keys & 2 addtional & 2 markers at head & middle",
    487                 new String[] { "%", "A", "%", "B" },
    488                 new String[] { "1", "2" },
    489                 new String[] { "1", "A", "2", "B" });
    490         assertInsertAdditionalMoreKeys("2 more keys & 2 addtional & 2 markers at head & tail",
    491                 new String[] { "%", "A", "B", "%" },
    492                 new String[] { "1", "2" },
    493                 new String[] { "1", "A", "B", "2" });
    494         assertInsertAdditionalMoreKeys("2 more keys & 2 addtional & 2 markers at middle & tail",
    495                 new String[] { "A", "%", "B", "%" },
    496                 new String[] { "1", "2" },
    497                 new String[] { "A", "1", "B", "2" });
    498 
    499         // 2 markers & excess additional more keys.
    500         assertInsertAdditionalMoreKeys("0 more key & 2 additons & 2 markers",
    501                 new String[] { "%", "%" },
    502                 new String[] { "1", "2", "3" },
    503                 new String[] { "1", "2", "3" });
    504         assertInsertAdditionalMoreKeys("1 more key & 2 additons & 2 markers at head",
    505                 new String[] { "%", "%", "A" },
    506                 new String[] { "1", "2", "3" },
    507                 new String[] { "1", "2", "A", "3" });
    508         assertInsertAdditionalMoreKeys("1 more key & 2 additons & 2 markers at tail",
    509                 new String[] { "A", "%", "%" },
    510                 new String[] { "1", "2", "3" },
    511                 new String[] { "A", "1", "2", "3" });
    512         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & 2 markers at middle",
    513                 new String[] { "A", "%", "%", "B" },
    514                 new String[] { "1", "2", "3" },
    515                 new String[] { "A", "1", "2", "B", "3" });
    516         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & 2 markers at head & middle",
    517                 new String[] { "%", "A", "%", "B" },
    518                 new String[] { "1", "2", "3" },
    519                 new String[] { "1", "A", "2", "B", "3" });
    520         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & 2 markers at head & tail",
    521                 new String[] { "%", "A", "B", "%" },
    522                 new String[] { "1", "2", "3" },
    523                 new String[] { "1", "A", "B", "2", "3" });
    524         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & 2 markers at middle & tail",
    525                 new String[] { "A", "%", "B", "%" },
    526                 new String[] { "1", "2", "3" },
    527                 new String[] { "A", "1", "B", "2", "3" });
    528 
    529         // 0 addtional more key and excess markers.
    530         assertInsertAdditionalMoreKeys("0 more key & null & excess marker",
    531                 new String[] { "%" },
    532                 null,
    533                 null);
    534         assertInsertAdditionalMoreKeys("1 more key & null & excess marker at head",
    535                 new String[] { "%", "A" },
    536                 null,
    537                 new String[] { "A" });
    538         assertInsertAdditionalMoreKeys("1 more key & null & excess marker at tail",
    539                 new String[] { "A", "%" },
    540                 null,
    541                 new String[] { "A" });
    542         assertInsertAdditionalMoreKeys("2 more keys & null & excess marker at middle",
    543                 new String[] { "A", "%", "B" },
    544                 null,
    545                 new String[] { "A", "B" });
    546         assertInsertAdditionalMoreKeys("2 more keys & null & excess markers",
    547                 new String[] { "%", "A", "%", "B", "%" },
    548                 null,
    549                 new String[] { "A", "B" });
    550 
    551         // Excess markers.
    552         assertInsertAdditionalMoreKeys("0 more key & 1 additon & excess marker",
    553                 new String[] { "%", "%" },
    554                 new String[] { "1" },
    555                 new String[] { "1" });
    556         assertInsertAdditionalMoreKeys("1 more key & 1 additon & excess marker at head",
    557                 new String[] { "%", "%", "A" },
    558                 new String[] { "1" },
    559                 new String[] { "1", "A" });
    560         assertInsertAdditionalMoreKeys("1 more key & 1 additon & excess marker at tail",
    561                 new String[] { "A", "%", "%" },
    562                 new String[] { "1" },
    563                 new String[] { "A", "1" });
    564         assertInsertAdditionalMoreKeys("2 more keys & 1 additon & excess marker at middle",
    565                 new String[] { "A", "%", "%", "B" },
    566                 new String[] { "1" },
    567                 new String[] { "A", "1", "B" });
    568         assertInsertAdditionalMoreKeys("2 more keys & 1 additon & excess markers",
    569                 new String[] { "%", "A", "%", "B", "%" },
    570                 new String[] { "1" },
    571                 new String[] { "1", "A", "B" });
    572         assertInsertAdditionalMoreKeys("2 more keys & 2 additons & excess markers",
    573                 new String[] { "%", "A", "%", "B", "%" },
    574                 new String[] { "1", "2" },
    575                 new String[] { "1", "A", "2", "B" });
    576         assertInsertAdditionalMoreKeys("2 more keys & 3 additons & excess markers",
    577                 new String[] { "%", "A", "%", "%", "B", "%" },
    578                 new String[] { "1", "2", "3" },
    579                 new String[] { "1", "A", "2", "3", "B" });
    580     }
    581 
    582     private static final String HAS_LABEL = "!hasLabel!";
    583     private static final String NEEDS_DIVIDER = "!needsDividers!";
    584     private static final String AUTO_COLUMN_ORDER = "!autoColumnOrder!";
    585     private static final String FIXED_COLUMN_ORDER = "!fixedColumnOrder!";
    586 
    587     private static void assertGetBooleanValue(String message, String key, String[] moreKeys,
    588             String[] expected, boolean expectedValue) {
    589         final String[] actual = Arrays.copyOf(moreKeys, moreKeys.length);
    590         final boolean actualValue = KeySpecParser.getBooleanValue(actual, key);
    591         assertEquals(message + " [value]", expectedValue, actualValue);
    592         assertArrayEquals(message, expected, actual);
    593     }
    594 
    595     public void testGetBooleanValue() {
    596         assertGetBooleanValue("Has label", HAS_LABEL,
    597                 new String[] { HAS_LABEL, "a", "b", "c" },
    598                 new String[] { null, "a", "b", "c" }, true);
    599         // Upper case specification will not work.
    600         assertGetBooleanValue("HAS LABEL", HAS_LABEL,
    601                 new String[] { HAS_LABEL.toUpperCase(Locale.ROOT), "a", "b", "c" },
    602                 new String[] { "!HASLABEL!", "a", "b", "c" }, false);
    603 
    604         assertGetBooleanValue("No has label", HAS_LABEL,
    605                 new String[] { "a", "b", "c" },
    606                 new String[] { "a", "b", "c" }, false);
    607         assertGetBooleanValue("No has label with fixed clumn order", HAS_LABEL,
    608                 new String[] { FIXED_COLUMN_ORDER + "3", "a", "b", "c" },
    609                 new String[] { FIXED_COLUMN_ORDER + "3", "a", "b", "c" }, false);
    610 
    611         // Upper case specification will not work.
    612         assertGetBooleanValue("Multiple has label", HAS_LABEL,
    613                 new String[] {
    614                     "a", HAS_LABEL.toUpperCase(Locale.ROOT), "b", "c", HAS_LABEL, "d" },
    615                 new String[] {
    616                     "a", "!HASLABEL!", "b", "c", null, "d" }, true);
    617         // Upper case specification will not work.
    618         assertGetBooleanValue("Multiple has label with needs dividers", HAS_LABEL,
    619                 new String[] {
    620                     "a", HAS_LABEL, "b", NEEDS_DIVIDER, HAS_LABEL.toUpperCase(Locale.ROOT), "d" },
    621                 new String[] {
    622                     "a", null, "b", NEEDS_DIVIDER, "!HASLABEL!", "d" }, true);
    623     }
    624 
    625     private static void assertGetIntValue(String message, String key, int defaultValue,
    626             String[] moreKeys, String[] expected, int expectedValue) {
    627         final String[] actual = Arrays.copyOf(moreKeys, moreKeys.length);
    628         final int actualValue = KeySpecParser.getIntValue(actual, key, defaultValue);
    629         assertEquals(message + " [value]", expectedValue, actualValue);
    630         assertArrayEquals(message, expected, actual);
    631     }
    632 
    633     public void testGetIntValue() {
    634         assertGetIntValue("Fixed column order 3", FIXED_COLUMN_ORDER, -1,
    635                 new String[] { FIXED_COLUMN_ORDER + "3", "a", "b", "c" },
    636                 new String[] { null, "a", "b", "c" }, 3);
    637         // Upper case specification will not work.
    638         assertGetIntValue("FIXED COLUMN ORDER 3", FIXED_COLUMN_ORDER, -1,
    639                 new String[] { FIXED_COLUMN_ORDER.toUpperCase(Locale.ROOT) + "3", "a", "b", "c" },
    640                 new String[] { "!FIXEDCOLUMNORDER!3", "a", "b", "c" }, -1);
    641 
    642         assertGetIntValue("No fixed column order", FIXED_COLUMN_ORDER, -1,
    643                 new String[] { "a", "b", "c" },
    644                 new String[] { "a", "b", "c" }, -1);
    645         assertGetIntValue("No fixed column order with auto column order", FIXED_COLUMN_ORDER, -1,
    646                 new String[] { AUTO_COLUMN_ORDER + "5", "a", "b", "c" },
    647                 new String[] { AUTO_COLUMN_ORDER + "5", "a", "b", "c" }, -1);
    648 
    649         assertGetIntValue("Multiple fixed column order 3,5", FIXED_COLUMN_ORDER, -1,
    650                 new String[] { FIXED_COLUMN_ORDER + "3", "a", FIXED_COLUMN_ORDER + "5", "b" },
    651                 new String[] { null, "a", null, "b" }, 3);
    652         // Upper case specification will not work.
    653         assertGetIntValue("Multiple fixed column order 5,3 with has label", FIXED_COLUMN_ORDER, -1,
    654                 new String[] {
    655                     FIXED_COLUMN_ORDER.toUpperCase(Locale.ROOT) + "5", HAS_LABEL, "a",
    656                     FIXED_COLUMN_ORDER + "3", "b" },
    657                 new String[] { "!FIXEDCOLUMNORDER!5", HAS_LABEL, "a", null, "b" }, 3);
    658     }
    659 }
    660