aboutsummaryrefslogtreecommitdiff
path: root/libxsde/xsde/cxx/serializer/validating/string-common.cxx
blob: f19caade1ad94e33fedf42339d734e000e587575 (plain)
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
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// file      : xsde/cxx/serializer/validating/string-common.cxx
// copyright : Copyright (c) 2005-2011 Code Synthesis Tools CC
// license   : GNU GPL v2 + exceptions; see accompanying LICENSE file

#include <xsde/cxx/config.hxx>

#include <string.h> // strlen

#ifdef XSDE_REGEXP
#ifdef XSDE_EXCEPTIONS
#  include <new> // std::bad_alloc
#endif
#endif

#include <xsde/cxx/string-search.hxx>
#include <xsde/cxx/serializer/validating/string-common.hxx>

namespace xsde
{
  namespace cxx
  {
    namespace serializer
    {
      namespace validating
      {
        bool string_common::
        validate_facets (const char* s,
                         const string_facets::facets& f,
                         context& ctx)
        {
          if (f.length_set_ ||
              f.min_length_set_ ||
              f.max_length_set_ ||
              f.enum_count_ != 0
#ifdef XSDE_REGEXP
              || f.pattern_set_ != 0
#endif
          )
          {
            return validate_facets (s, strlen (s), f, ctx);
          }

          return true;
        }

        bool string_common::
        validate_facets (const char* s,
                         size_t n,
                         const string_facets::facets& f,
                         context& ctx)
        {
          if (f.length_set_ && n != f.length_)
          {
            ctx.schema_error (schema_error::length_not_equal_prescribed);
            return false;
          }

          if (f.min_length_set_ && n < f.min_length_)
          {
            ctx.schema_error (schema_error::length_less_than_min);
            return false;
          }

          if (f.max_length_set_ && n > f.max_length_)
          {
            ctx.schema_error (schema_error::length_greater_than_max);
            return false;
          }

          if (f.enum_count_ != 0)
          {
            if (search (f.enum_, f.enum_count_, s) == f.enum_count_)
            {
              ctx.schema_error (schema_error::value_not_in_enumeration);
              return false;
            }
          }

#ifdef XSDE_REGEXP
          if (f.pattern_set_ != 0)
          {
            if (f.pattern_set_ == 1)
            {
              xmlRegexpPtr r = xmlRegexpCompile (
                reinterpret_cast<const xmlChar*> (f.pattern_.str));

              if (r == 0)
              {
#ifdef XSDE_EXCEPTIONS
                throw std::bad_alloc ();
#else
                ctx.sys_error (sys_error::no_memory);
                return false;
#endif

              }

              string_facets::facets& t =
                const_cast<string_facets::facets&> (f);

              t.pattern_.regexp = r;
              t.pattern_set_ = 2;
            }

            if (xmlRegexpExec (
                  f.pattern_.regexp,
                  reinterpret_cast<const xmlChar*> (s)) != 1)
            {
              ctx.schema_error (schema_error::value_pattern_mismatch);
              return false;
            }
          }
#endif
          return true;
        }
      }
    }
  }
}