1 /*
2  * Copyright (c) 2012, 2016, Oracle and/or its affiliates. All rights reserved.
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * This code is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License version 2 only, as
7  * published by the Free Software Foundation.
8  *
9  * This code is distributed in the hope that it will be useful, but WITHOUT
10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12  * version 2 for more details (a copy is included in the LICENSE file that
13  * accompanied this code).
14  *
15  * You should have received a copy of the GNU General Public License version
16  * 2 along with this work; if not, write to the Free Software Foundation,
17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18  *
19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20  * or visit www.oracle.com if you need additional information or have any
21  * questions.
22  */
23 
24 /*
25  * This file is available under and governed by the GNU General Public
26  * License version 2 only, as published by the Free Software Foundation.
27  * However, the following notice accompanied the original version of this
28  * file:
29  *
30  * Copyright (c) 2008-2012, Stephen Colebourne & Michael Nascimento Santos
31  *
32  * All rights reserved.
33  *
34  * Redistribution and use in source and binary forms, with or without
35  * modification, are permitted provided that the following conditions are met:
36  *
37  *  * Redistributions of source code must retain the above copyright notice,
38  *    this list of conditions and the following disclaimer.
39  *
40  *  * Redistributions in binary form must reproduce the above copyright notice,
41  *    this list of conditions and the following disclaimer in the documentation
42  *    and/or other materials provided with the distribution.
43  *
44  *  * Neither the name of JSR-310 nor the names of its contributors
45  *    may be used to endorse or promote products derived from this software
46  *    without specific prior written permission.
47  *
48  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
49  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
50  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
51  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
52  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
53  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
54  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
55  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
56  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
57  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
58  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
59  */
60 
61 /*
62  * @test
63  * @modules jdk.localedata
64  */
65 
66 package test.java.time.format;
67 
68 import java.text.ParsePosition;
69 import java.time.chrono.ChronoLocalDate;
70 import java.time.chrono.JapaneseChronology;
71 import java.time.chrono.HijrahDate;
72 import java.time.chrono.JapaneseDate;
73 import java.time.chrono.MinguoDate;
74 import java.time.chrono.ThaiBuddhistDate;
75 import java.time.DayOfWeek;
76 import java.time.format.DateTimeFormatter;
77 import java.time.format.DateTimeFormatterBuilder;
78 import java.time.format.TextStyle;
79 import java.time.format.SignStyle;
80 import java.time.temporal.ChronoField;
81 import java.time.temporal.TemporalField;
82 import java.util.Locale;
83 
84 import org.testng.annotations.DataProvider;
85 import org.testng.annotations.Test;
86 
87 
88 import static java.time.temporal.ChronoField.DAY_OF_MONTH;
89 import static java.time.temporal.ChronoField.DAY_OF_WEEK;
90 import static java.time.temporal.ChronoField.MONTH_OF_YEAR;
91 import static org.testng.Assert.assertEquals;
92 
93 /**
94  * Test TextPrinterParser.
95  */
96 @Test
97 public class TestTextParserWithLocale extends AbstractTestPrinterParser {
98     static final Locale RUSSIAN = new Locale("ru");
99     static final Locale FINNISH = new Locale("fi");
100 
101     @DataProvider(name="parseDayOfWeekText")
providerDayOfWeekData()102     Object[][] providerDayOfWeekData() {
103         return new Object[][] {
104             // Locale, pattern, input text, expected DayOfWeek
105             {Locale.US, "e",  "1",  DayOfWeek.SUNDAY},
106             {Locale.US, "ee", "01", DayOfWeek.SUNDAY},
107             {Locale.US, "c",  "1",  DayOfWeek.SUNDAY},
108 
109             {Locale.UK, "e",  "1",  DayOfWeek.MONDAY},
110             {Locale.UK, "ee", "01", DayOfWeek.MONDAY},
111             {Locale.UK, "c",  "1",  DayOfWeek.MONDAY},
112         };
113     }
114 
115     @Test(dataProvider="parseDayOfWeekText")
test_parseDayOfWeekText(Locale locale, String pattern, String input, DayOfWeek expected)116     public void test_parseDayOfWeekText(Locale locale, String pattern, String input, DayOfWeek expected) {
117         DateTimeFormatter formatter = getPatternFormatter(pattern).withLocale(locale);
118         ParsePosition pos = new ParsePosition(0);
119         assertEquals(DayOfWeek.from(formatter.parse(input, pos)), expected);
120         assertEquals(pos.getIndex(), input.length());
121     }
122 
123     //--------------------------------------------------------------------
124     // Test data is dependent on localized resources.
125     @DataProvider(name="parseStandaloneText")
providerStandaloneText()126     Object[][] providerStandaloneText() {
127         // Locale, TemporalField, TextStyle, expected value, input text
128         return new Object[][] {
129             {RUSSIAN, MONTH_OF_YEAR, TextStyle.FULL_STANDALONE,   1, "\u044f\u043d\u0432\u0430\u0440\u044c"},
130             {RUSSIAN, MONTH_OF_YEAR, TextStyle.FULL_STANDALONE,  12, "\u0434\u0435\u043a\u0430\u0431\u0440\u044c"},
131             {RUSSIAN, MONTH_OF_YEAR, TextStyle.SHORT_STANDALONE,  1, "\u044f\u043d\u0432."},
132             {RUSSIAN, MONTH_OF_YEAR, TextStyle.SHORT_STANDALONE, 12, "\u0434\u0435\u043a."},
133             {FINNISH, DAY_OF_WEEK,   TextStyle.FULL_STANDALONE,   2, "tiistai"},
134             {FINNISH, DAY_OF_WEEK,   TextStyle.SHORT_STANDALONE,  2, "ti"},
135         };
136     }
137 
138     // Test data is dependent on localized resources.
139     @DataProvider(name="parseLenientText")
providerLenientText()140     Object[][] providerLenientText() {
141         // Locale, TemporalField, expected value, input text
142         return new Object[][] {
143             {RUSSIAN, MONTH_OF_YEAR, 1, "\u044f\u043d\u0432\u0430\u0440\u044f"}, // full format
144             {RUSSIAN, MONTH_OF_YEAR, 1, "\u044f\u043d\u0432\u0430\u0440\u044c"}, // full standalone
145             {RUSSIAN, MONTH_OF_YEAR, 1, "\u044f\u043d\u0432."}, // short format
146             {RUSSIAN, MONTH_OF_YEAR, 1, "\u044f\u043d\u0432."}, // short standalone
147         };
148     }
149 
150     @Test(dataProvider="parseStandaloneText")
test_parseStandaloneText(Locale locale, TemporalField field, TextStyle style, int expectedValue, String input)151     public void test_parseStandaloneText(Locale locale, TemporalField field, TextStyle style, int expectedValue, String input) {
152         DateTimeFormatter formatter = getFormatter(field, style).withLocale(locale);
153         ParsePosition pos = new ParsePosition(0);
154         assertEquals(formatter.parseUnresolved(input, pos).getLong(field), (long) expectedValue);
155         assertEquals(pos.getIndex(), input.length());
156     }
157 
158     //-----------------------------------------------------------------------
test_parse_french_short_strict_full_noMatch()159     public void test_parse_french_short_strict_full_noMatch() throws Exception {
160         setStrict(true);
161         ParsePosition pos = new ParsePosition(0);
162         getFormatter(MONTH_OF_YEAR, TextStyle.SHORT).withLocale(Locale.FRENCH)
163                                                     .parseUnresolved("janvier", pos);
164         assertEquals(pos.getErrorIndex(), 0);
165     }
166 
test_parse_french_short_strict_short_match()167     public void test_parse_french_short_strict_short_match() throws Exception {
168         setStrict(true);
169         ParsePosition pos = new ParsePosition(0);
170         assertEquals(getFormatter(MONTH_OF_YEAR, TextStyle.SHORT).withLocale(Locale.FRENCH)
171                                                                  .parseUnresolved("janv.", pos)
172                                                                  .getLong(MONTH_OF_YEAR),
173                      1L);
174         assertEquals(pos.getIndex(), 5);
175     }
176 
177     //-----------------------------------------------------------------------
178 
179     @Test(dataProvider="parseLenientText")
test_parseLenientText(Locale locale, TemporalField field, int expectedValue, String input)180     public void test_parseLenientText(Locale locale, TemporalField field, int expectedValue, String input) {
181         setStrict(false);
182         ParsePosition pos = new ParsePosition(0);
183         DateTimeFormatter formatter = getFormatter(field).withLocale(locale);
184         assertEquals(formatter.parseUnresolved(input, pos).getLong(field), (long) expectedValue);
185         assertEquals(pos.getIndex(), input.length());
186     }
187 
188 
189     //-----------------------------------------------------------------------
190     @DataProvider(name="parseChronoLocalDate")
provider_chronoLocalDate()191     Object[][] provider_chronoLocalDate() {
192         return new Object[][] {
193             { HijrahDate.now() },
194             { JapaneseDate.now() },
195             { MinguoDate.now() },
196             { ThaiBuddhistDate.now() }};
197     }
198 
199     private static final DateTimeFormatter fmt_chrono =
200         new DateTimeFormatterBuilder()
201             .optionalStart()
202             .appendChronologyId()
203             .appendLiteral(' ')
204             .optionalEnd()
205             .optionalStart()
206             .appendText(ChronoField.ERA, TextStyle.SHORT)
207             .appendLiteral(' ')
208             .optionalEnd()
209             .appendValue(ChronoField.YEAR_OF_ERA, 1, 9, SignStyle.NORMAL)
210             .appendLiteral('-')
211             .appendValue(ChronoField.MONTH_OF_YEAR, 1, 2, SignStyle.NEVER)
212             .appendLiteral('-')
213             .appendValue(ChronoField.DAY_OF_MONTH, 1, 2, SignStyle.NEVER)
214             .toFormatter();
215 
216     @Test(dataProvider="parseChronoLocalDate")
test_chronoLocalDate(ChronoLocalDate date)217     public void test_chronoLocalDate(ChronoLocalDate date) throws Exception {
218         System.out.printf(" %s, [fmt=%s]%n", date, fmt_chrono.format(date));
219         assertEquals(date, fmt_chrono.parse(fmt_chrono.format(date), ChronoLocalDate::from));
220 
221         DateTimeFormatter fmt = DateTimeFormatter.ofPattern("[GGG ]yyy-MM-dd")
222                                                  .withChronology(date.getChronology());
223         System.out.printf(" %s, [fmt=%s]%n", date.toString(), fmt.format(date));
224         assertEquals(date, fmt.parse(fmt.format(date), ChronoLocalDate::from));
225     }
226 }
227