Passed
Pull Request — 4.0 (#7796)
by Roman
12:08 queued 05:44
created

DBDateTest::testInvertedYearCorrection()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 7
Code Lines 4

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
eloc 4
nc 1
nop 0
dl 0
loc 7
rs 9.4285
c 0
b 0
f 0
1
<?php
2
3
namespace SilverStripe\ORM\Tests;
4
5
use SilverStripe\Dev\SapphireTest;
6
use SilverStripe\i18n\i18n;
7
use SilverStripe\ORM\FieldType\DBDate;
8
use SilverStripe\ORM\FieldType\DBDatetime;
9
use SilverStripe\ORM\FieldType\DBField;
10
11
/**
12
 * @skipUpgrade
13
 */
14
class DBDateTest extends SapphireTest
15
{
16
    protected $oldError = null;
17
18
    protected function setUp()
19
    {
20
        parent::setUp();
21
        $this->oldError = error_reporting();
22
        // Validate setup
23
        assert(date_default_timezone_get() === 'UTC');
24
        i18n::set_locale('en_NZ');
25
    }
26
27
    protected function tearDown()
28
    {
29
        $this->restoreNotices();
30
        parent::tearDown();
31
    }
32
33
    /**
34
     * Temporarily disable notices
35
     */
36
    protected function suppressNotices()
37
    {
38
        error_reporting(error_reporting() & ~E_USER_NOTICE);
39
        \PHPUnit_Framework_Error_Notice::$enabled = false;
40
    }
41
42
    /**
43
     * Restore notices
44
     */
45
    protected function restoreNotices()
46
    {
47
        error_reporting($this->oldError);
48
        \PHPUnit_Framework_Error_Notice::$enabled = true;
49
    }
50
51
    public function testNiceDate()
52
    {
53
        $this->assertEquals(
54
            '31/03/2008',
55
            DBField::create_field('Date', 1206968400)->Nice(),
0 ignored issues
show
Bug introduced by
The method Nice() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

55
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ Nice(),
Loading history...
56
            "Date->Nice() works with timestamp integers"
57
        );
58
        $this->assertEquals(
59
            '30/03/2008',
60
            DBField::create_field('Date', 1206882000)->Nice(),
61
            "Date->Nice() works with timestamp integers"
62
        );
63
        $this->assertEquals(
64
            '31/03/2008',
65
            DBField::create_field('Date', '1206968400')->Nice(),
66
            "Date->Nice() works with timestamp strings"
67
        );
68
        $this->assertEquals(
69
            '30/03/2008',
70
            DBField::create_field('Date', '1206882000')->Nice(),
71
            "Date->Nice() works with timestamp strings"
72
        );
73
        $this->assertEquals(
74
            '4/03/2003',
75
            DBField::create_field('Date', '4.3.2003')->Nice(),
76
            "Date->Nice() works with D.M.YYYY format"
77
        );
78
        $this->assertEquals(
79
            '4/03/2003',
80
            DBField::create_field('Date', '04.03.2003')->Nice(),
81
            "Date->Nice() works with DD.MM.YYYY format"
82
        );
83
        $this->assertEquals(
84
            '4/03/2003',
85
            DBField::create_field('Date', '2003-3-4')->Nice(),
86
            "Date->Nice() works with YYYY-M-D format"
87
        );
88
        $this->assertEquals(
89
            '4/03/2003',
90
            DBField::create_field('Date', '2003-03-04')->Nice(),
91
            "Date->Nice() works with YYYY-MM-DD format"
92
        );
93
    }
94
95
    /**
96
     * @expectedException \InvalidArgumentException
97
     * @expectedExceptionMessage Invalid date: '3/16/2003'. Use y-MM-dd to prevent this error.
98
     */
99
    public function testMDYConversion()
100
    {
101
        DBField::create_field('Date', '3/16/2003');
102
    }
103
104
    /**
105
     * @expectedException \InvalidArgumentException
106
     * @expectedExceptionMessage Invalid date: '03-03-04'. Use y-MM-dd to prevent this error.
107
     */
108
    public function testY2kCorrection()
109
    {
110
        DBField::create_field('Date', '03-03-04');
111
    }
112
113
    public function testInvertedYearCorrection()
114
    {
115
        // iso8601 expects year first, but support year last
116
        $this->assertEquals(
117
            '4/03/2003',
118
            DBField::create_field('Date', '04-03-2003')->Nice(),
119
            "Date->Nice() works with DD-MM-YYYY format"
120
        );
121
    }
122
123
    public function testYear()
124
    {
125
        $this->assertEquals(
126
            '2008',
127
            DBField::create_field('Date', 1206968400)->Year(),
0 ignored issues
show
Bug introduced by
The method Year() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

127
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ Year(),
Loading history...
128
            "Date->Year() works with timestamp integers"
129
        );
130
    }
131
132
    public function testDayOfWeek()
133
    {
134
        $this->assertEquals(
135
            'Monday',
136
            DBField::create_field('Date', 1206968400)->DayOfWeek(),
0 ignored issues
show
Bug introduced by
The method DayOfWeek() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

136
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ DayOfWeek(),
Loading history...
137
            "Date->Day() works with timestamp integers"
138
        );
139
    }
140
141
    public function testMonth()
142
    {
143
        $this->assertEquals(
144
            'March',
145
            DBField::create_field('Date', 1206968400)->Month(),
0 ignored issues
show
Bug introduced by
The method Month() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

145
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ Month(),
Loading history...
146
            "Date->Month() works with timestamp integers"
147
        );
148
    }
149
150
    public function testShortMonth()
151
    {
152
        $this->assertEquals(
153
            'Mar',
154
            DBField::create_field('Date', 1206968400)->ShortMonth(),
0 ignored issues
show
Bug introduced by
The method ShortMonth() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

154
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ ShortMonth(),
Loading history...
155
            "Date->ShortMonth() works with timestamp integers"
156
        );
157
    }
158
159
    public function testLongDate()
160
    {
161
        $this->assertEquals(
162
            '31 March 2008',
163
            DBField::create_field('Date', 1206968400)->Long(),
0 ignored issues
show
Bug introduced by
The method Long() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

163
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ Long(),
Loading history...
164
            "Date->Long() works with numeric timestamp"
165
        );
166
        $this->assertEquals(
167
            '31 March 2008',
168
            DBField::create_field('Date', '1206968400')->Long(),
169
            "Date->Long() works with string timestamp"
170
        );
171
        $this->assertEquals(
172
            '30 March 2008',
173
            DBField::create_field('Date', 1206882000)->Long(),
174
            "Date->Long() works with numeric timestamp"
175
        );
176
        $this->assertEquals(
177
            '30 March 2008',
178
            DBField::create_field('Date', '1206882000')->Long(),
179
            "Date->Long() works with numeric timestamp"
180
        );
181
        $this->assertEquals(
182
            '3 April 2003',
183
            DBField::create_field('Date', '2003-4-3')->Long(),
184
            "Date->Long() works with YYYY-M-D"
185
        );
186
        $this->assertEquals(
187
            '3 April 2003',
188
            DBField::create_field('Date', '3.4.2003')->Long(),
189
            "Date->Long() works with D.M.YYYY"
190
        );
191
    }
192
193
    public function testFull()
194
    {
195
        $this->assertEquals(
196
            'Monday, 31 March 2008',
197
            DBField::create_field('Date', 1206968400)->Full(),
0 ignored issues
show
Bug introduced by
The method Full() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

197
            DBField::create_field('Date', 1206968400)->/** @scrutinizer ignore-call */ Full(),
Loading history...
198
            "Date->Full() works with timestamp integers"
199
        );
200
    }
201
202
    public function testSetNullAndZeroValues()
203
    {
204
        $date = DBField::create_field('Date', '');
205
        $this->assertNull($date->getValue(), 'Empty string evaluates to NULL');
206
207
        $date = DBField::create_field('Date', null);
208
        $this->assertNull($date->getValue(), 'NULL is set as NULL');
209
210
        $date = DBField::create_field('Date', false);
211
        $this->assertNull($date->getValue(), 'Boolean FALSE evaluates to NULL');
212
213
        $date = DBField::create_field('Date', array());
214
        $this->assertNull($date->getValue(), 'Empty array evaluates to NULL');
215
216
        $date = DBField::create_field('Date', '0');
217
        $this->assertEquals('1970-01-01', $date->getValue(), 'Zero is UNIX epoch date');
218
219
        $date = DBField::create_field('Date', 0);
220
        $this->assertEquals('1970-01-01', $date->getValue(), 'Zero is UNIX epoch date');
221
222
        $date = DBField::create_field('Date', '0000-00-00 00:00:00');
223
        $this->assertNull($date->getValue(), '0000-00-00 00:00:00 is set as NULL');
224
225
        $date = DBField::create_field('Date', '00/00/0000');
226
        $this->assertNull($date->getValue(), '00/00/0000 is set as NULL');
227
    }
228
229
    public function testDayOfMonth()
230
    {
231
        $date = DBField::create_field('Date', '2000-10-10');
232
        $this->assertEquals('10', $date->DayOfMonth());
0 ignored issues
show
Bug introduced by
The method DayOfMonth() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

232
        $this->assertEquals('10', $date->/** @scrutinizer ignore-call */ DayOfMonth());
Loading history...
233
        $this->assertEquals('10th', $date->DayOfMonth(true));
234
235
        $range = $date->RangeString(DBField::create_field('Date', '2000-10-20'));
0 ignored issues
show
Bug introduced by
The method RangeString() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

235
        /** @scrutinizer ignore-call */ 
236
        $range = $date->RangeString(DBField::create_field('Date', '2000-10-20'));
Loading history...
236
        $this->assertEquals('10 - 20 Oct 2000', $range);
237
        $range = $date->RangeString(DBField::create_field('Date', '2000-10-20'), true);
238
        $this->assertEquals('10th - 20th Oct 2000', $range);
239
    }
240
241
    public function testExtendedDates()
242
    {
243
        $date = DBField::create_field('Date', '1800-10-10');
244
        $this->assertEquals('10 Oct 1800', $date->Format('dd MMM y'));
0 ignored issues
show
Bug introduced by
The method Format() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

244
        $this->assertEquals('10 Oct 1800', $date->/** @scrutinizer ignore-call */ Format('dd MMM y'));
Loading history...
245
246
        // Note: Fails around 1500 or older
247
        $date = DBField::create_field('Date', '1600-10-10');
248
        $this->assertEquals('10 Oct 1600', $date->Format('dd MMM y'));
249
250
        $date = DBField::create_field('Date', '3000-4-3');
251
        $this->assertEquals('03 Apr 3000', $date->Format('dd MMM y'));
252
    }
253
254
    public function testAgoInPast()
255
    {
256
        DBDatetime::set_mock_now('2000-12-31 12:00:00');
257
258
        $this->assertEquals(
259
            '1 month ago',
260
            DBField::create_field('Date', '2000-11-26')->Ago(true, 1),
0 ignored issues
show
Bug introduced by
The method Ago() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

260
            DBField::create_field('Date', '2000-11-26')->/** @scrutinizer ignore-call */ Ago(true, 1),
Loading history...
261
            'Past match on days, less than two months, lowest significance'
262
        );
263
264
        $this->assertEquals(
265
            '50 days ago', // Rounded from 49.5 days up
266
            DBField::create_field('Date', '2000-11-12')->Ago(),
267
            'Past match on days, less than two months'
268
        );
269
270
        $this->assertEquals(
271
            '2 months ago',
272
            DBField::create_field('Date', '2000-10-27')->Ago(),
273
            'Past match on days, over two months'
274
        );
275
276
        $this->assertEquals(
277
            '66 days ago', // rounded from 65.5 days up
278
            DBField::create_field('Date', '2000-10-27')->Ago(true, 3),
279
            'Past match on days, over two months, significance of 3'
280
        );
281
282
        $this->assertEquals(
283
            '10 years ago',
284
            DBField::create_field('Date', '1990-12-31')->Ago(),
285
            'Exact past match on years'
286
        );
287
288
        $this->assertEquals(
289
            '10 years ago',
290
            DBField::create_field('Date', '1990-12-30')->Ago(),
291
            'Approximate past match on years'
292
        );
293
294
        $this->assertEquals(
295
            '1 year ago',
296
            DBField::create_field('Date', '1999-12-30')->Ago(true, 1),
297
            'Approximate past match in singular, lowest significance'
298
        );
299
300
        $this->assertEquals(
301
            '12 months ago',
302
            DBField::create_field('Date', '1999-12-30')->Ago(),
303
            'Approximate past match in singular'
304
        );
305
306
        DBDatetime::clear_mock_now();
307
    }
308
309
    public function testAgoInFuture()
310
    {
311
        DBDatetime::set_mock_now('2000-12-31 00:00:00');
312
313
        $this->assertEquals(
314
            'in 10 years',
315
            DBField::create_field('Date', '2010-12-31')->Ago(),
316
            'Exact past match on years'
317
        );
318
319
        $this->assertEquals(
320
            'in 1 day',
321
            DBField::create_field('Date', '2001-01-01')->Ago(true, 1),
322
            'Approximate past match on minutes'
323
        );
324
325
        $this->assertEquals(
326
            'in 24 hours',
327
            DBField::create_field('Date', '2001-01-01')->Ago(),
328
            'Approximate past match on minutes'
329
        );
330
331
        DBDatetime::clear_mock_now();
332
    }
333
334
    public function testRfc3999()
335
    {
336
        // Dates should be formatted as: 2018-01-24T14:05:53+00:00
0 ignored issues
show
Unused Code Comprehensibility introduced by
44% of this comment could be valid code. Did you maybe forget this after debugging?

Sometimes obsolete code just ends up commented out instead of removed. In this case it is better to remove the code once you have checked you do not need it.

The code might also have been commented out for debugging purposes. In this case it is vital that someone uncomments it again or your project may behave in very unexpected ways in production.

This check looks for comments that seem to be mostly valid code and reports them.

Loading history...
337
        $date = DBDate::create_field('Date', '2010-12-31');
338
        $this->assertEquals('2010-12-31T00:00:00+00:00', $date->Rfc3339());
0 ignored issues
show
Bug introduced by
The method Rfc3339() does not exist on SilverStripe\ORM\FieldType\DBField. Since you implemented __call, consider adding a @method annotation. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

338
        $this->assertEquals('2010-12-31T00:00:00+00:00', $date->/** @scrutinizer ignore-call */ Rfc3339());
Loading history...
339
    }
340
}
341