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
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
|
// file : xsde/cxx/serializer/validating/date.cxx
// license : GNU GPL v2 + exceptions; see accompanying LICENSE file
#include <stdio.h> // sprintf/snprintf
#include <xsde/cxx/serializer/validating/date.hxx>
#include <xsde/cxx/serializer/validating/time-zone.hxx>
namespace xsde
{
namespace cxx
{
namespace serializer
{
namespace validating
{
void date_simpl::
pre (const date& value)
{
value_ = value;
}
void date_simpl::
_serialize_content ()
{
// We only need strlen("-2147483649-MM-DD+hh:mm") + 1 characters to
// hold all representations of date.
//
char str[24];
int y = value_.year ();
unsigned short m = value_.month ();
unsigned short d = value_.day ();
// Validate day according to the XML Schema 1.1 specification:
//
// The day value must be no more than 30 if month is one of 4, 6, 9,
// or 11, no more than 28 if month is 2 and year is not divisible by
// 4, or is divisible by 100 but not by 400, and no more than 29 if
// month is 2 and year is divisible by 400, or by 4 but not by 100.
//
unsigned short max_day = 31;
switch (m)
{
case 4:
case 6:
case 9:
case 11:
max_day = 30;
break;
case 2:
max_day = ((y % 400 == 0) ||
(y % 4 == 0 && y % 100 != 0) ? 29 : 28);
break;
default:
break;
}
if (y != 0 && m > 0 && m < 13 && d > 0 && d <= max_day &&
(!value_.zone_present () || bits::valid_time_zone (value_)))
{
#ifdef XSDE_SNPRINTF
int n = snprintf (str, 18, "%.4d-%.2u-%.2u",
value_.year (), value_.month (), value_.day ());
#else
int n = sprintf (str, "%.4d-%.2u-%.2u",
value_.year (), value_.month (), value_.day ());
#endif
if (n > 0 && n < 18)
{
if (value_.zone_present ())
{
if (int z = bits::serialize_time_zone (str + n, value_))
n += z;
else
{
_schema_error (schema_error::invalid_date_value);
return;
}
}
_characters (str, static_cast<size_t> (n));
}
else
_schema_error (schema_error::invalid_date_value);
}
else
_schema_error (schema_error::invalid_date_value);
}
}
}
}
}
|