| 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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
 | # sql/functions.py
# Copyright (C) 2005-2012 the SQLAlchemy authors and contributors <see AUTHORS file>
#
# This module is part of SQLAlchemy and is released under
# the MIT License: http://www.opensource.org/licenses/mit-license.php
from .. import types as sqltypes, schema
from .expression import (
    ClauseList, Function, _literal_as_binds, literal_column, _type_from_args
    )
from . import operators
from .visitors import VisitableType
from .. import util
_registry = util.defaultdict(dict)
class _GenericMeta(VisitableType):
    def __init__(cls, clsname, bases, clsdict):
        cls.name = name = clsdict.get('name', clsname)
        package = clsdict.pop('package', '_default')
        # legacy
        if '__return_type__' in clsdict:
            cls.type = clsdict['__return_type__']
        reg = _registry[package]
        reg[name] = cls
        super(_GenericMeta, cls).__init__(clsname, bases, clsdict)
    def __call__(cls, *args, **kwargs):
        if cls.coerce_arguments:
            args = [_literal_as_binds(c) for c in args]
        return type.__call__(cls, *args, **kwargs)
class GenericFunction(Function):
    """Define a 'generic' function.
    A generic function is a pre-established :class:`.Function`
    class that is instantiated automatically when called
    by name from the :data:`.func` attribute.    Note that
    calling any name from :data:`.func` has the effect that
    a new :class:`.Function` instance is created automatically,
    given that name.  The primary use case for defining
    a :class:`.GenericFunction` class is so that a function
    of a particular name may be given a fixed return type.
    It can also include custom argument parsing schemes as well
    as additional methods.
    Subclasses of :class:`.GenericFunction` are automatically
    registered under the name of the class.  For
    example, a user-defined function ``as_utc()`` would
    be available immediately::
        from sqlalchemy.sql.functions import GenericFunction
        from sqlalchemy.types import DateTime
        class as_utc(GenericFunction):
            type = DateTime
        print select([func.as_utc()])
    User-defined generic functions can be organized into
    packages by specifying the "package" attribute when defining
    :class:`.GenericFunction`.   Third party libraries
    containing many functions may want to use this in order
    to avoid name conflicts with other systems.   For example,
    if our ``as_utc()`` function were part of a package
    "time"::
        class as_utc(GenericFunction):
            type = DateTime
            package = "time"
    The above function would be available from :data:`.func`
    using the package name ``time``::
        print select([func.time.as_utc()])
    .. versionadded:: 0.8 :class:`.GenericFunction` now supports
       automatic registration of new functions as well as package
       support.
    .. versionchanged:: 0.8 The attribute name ``type`` is used
       to specify the function's return type at the class level.
       Previously, the name ``__return_type__`` was used.  This
       name is still recognized for backwards-compatibility.
    """
    __metaclass__ = _GenericMeta
    coerce_arguments = True
    def __init__(self, *args, **kwargs):
        self.packagenames = []
        self._bind = kwargs.get('bind', None)
        self.clause_expr = ClauseList(
                operator=operators.comma_op,
                group_contents=True, *args).self_group()
        self.type = sqltypes.to_instance(
            kwargs.pop("type_", None) or getattr(self, 'type', None))
class next_value(GenericFunction):
    """Represent the 'next value', given a :class:`.Sequence`
    as it's single argument.
    Compiles into the appropriate function on each backend,
    or will raise NotImplementedError if used on a backend
    that does not provide support for sequences.
    """
    type = sqltypes.Integer()
    name = "next_value"
    coerce_arguments = False
    def __init__(self, seq, **kw):
        assert isinstance(seq, schema.Sequence), \
                "next_value() accepts a Sequence object as input."
        self._bind = kw.get('bind', None)
        self.sequence = seq
    @property
    def _from_objects(self):
        return []
class AnsiFunction(GenericFunction):
    def __init__(self, **kwargs):
        GenericFunction.__init__(self, **kwargs)
class ReturnTypeFromArgs(GenericFunction):
    """Define a function whose return type is the same as its arguments."""
    def __init__(self, *args, **kwargs):
        kwargs.setdefault('type_', _type_from_args(args))
        GenericFunction.__init__(self, *args, **kwargs)
class coalesce(ReturnTypeFromArgs):
    pass
class max(ReturnTypeFromArgs):
    pass
class min(ReturnTypeFromArgs):
    pass
class sum(ReturnTypeFromArgs):
    pass
class now(GenericFunction):
    type = sqltypes.DateTime
class concat(GenericFunction):
    type = sqltypes.String
class char_length(GenericFunction):
    type = sqltypes.Integer
    def __init__(self, arg, **kwargs):
        GenericFunction.__init__(self, arg, **kwargs)
class random(GenericFunction):
    pass
class count(GenericFunction):
    """The ANSI COUNT aggregate function.  With no arguments, emits COUNT \*."""
    type = sqltypes.Integer
    def __init__(self, expression=None, **kwargs):
        if expression is None:
            expression = literal_column('*')
        GenericFunction.__init__(self, expression, **kwargs)
class current_date(AnsiFunction):
    type = sqltypes.Date
class current_time(AnsiFunction):
    type = sqltypes.Time
class current_timestamp(AnsiFunction):
    type = sqltypes.DateTime
class current_user(AnsiFunction):
    type = sqltypes.String
class localtime(AnsiFunction):
    type = sqltypes.DateTime
class localtimestamp(AnsiFunction):
    type = sqltypes.DateTime
class session_user(AnsiFunction):
    type = sqltypes.String
class sysdate(AnsiFunction):
    type = sqltypes.DateTime
class user(AnsiFunction):
    type = sqltypes.String
 |