[tryton-debian-vcs] python-sql branch upstream updated. upstream/0.2-1-g5afdfac
Mathias Behrle
tryton-debian-vcs at alioth.debian.org
Fri Feb 28 16:49:38 UTC 2014
The following commit has been merged in the upstream branch:
https://alioth.debian.org/plugins/scmgit/cgi-bin/gitweb.cgi/?p=tryton/python-sql.git;a=commitdiff;h=upstream/0.2-1-g5afdfac
commit 5afdfac8ffaf9565e24bbcfbf3eb398bab946e91
Author: Mathias Behrle <mathiasb at m9s.biz>
Date: Fri Feb 28 17:25:57 2014 +0100
Adding upstream version 0.3.
diff --git a/CHANGELOG b/CHANGELOG
index 2780393..06d65f3 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,9 @@
+Version 0.3 - 2014-01-31
+* Add returning to Update
+* Add missing params for returning
+* Add columns definitions to Function
+* Fix AtTimeZone mapping
+
Version 0.2 - 2013-09-18
* Fix usage mixture between Div operator and function
* Add array support in operators
diff --git a/PKG-INFO b/PKG-INFO
index 2a7162e..1dd3522 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,6 +1,6 @@
Metadata-Version: 1.1
Name: python-sql
-Version: 0.2
+Version: 0.3
Summary: Library to write SQL queries
Home-page: http://code.google.com/p/python-sql/
Author: B2CK
diff --git a/python_sql.egg-info/PKG-INFO b/python_sql.egg-info/PKG-INFO
index 2a7162e..1dd3522 100644
--- a/python_sql.egg-info/PKG-INFO
+++ b/python_sql.egg-info/PKG-INFO
@@ -1,6 +1,6 @@
Metadata-Version: 1.1
Name: python-sql
-Version: 0.2
+Version: 0.3
Summary: Library to write SQL queries
Home-page: http://code.google.com/p/python-sql/
Author: B2CK
diff --git a/setup.py b/setup.py
index 28fe987..09e38fe 100644
--- a/setup.py
+++ b/setup.py
@@ -30,10 +30,12 @@
from setuptools import setup, find_packages
import os
import re
+import codecs
def read(fname):
- return open(os.path.join(os.path.dirname(__file__), fname)).read()
+ return codecs.open(
+ os.path.join(os.path.dirname(__file__), fname), 'r', 'utf-8').read()
def get_version():
diff --git a/sql/__init__.py b/sql/__init__.py
index 2c65f6b..1238546 100644
--- a/sql/__init__.py
+++ b/sql/__init__.py
@@ -29,7 +29,7 @@
from __future__ import division
-__version__ = '0.2'
+__version__ = '0.3'
__all__ = ['Flavor', 'Table', 'Literal', 'Column', 'Join', 'Asc', 'Desc']
import string
@@ -110,7 +110,7 @@ class AliasManager(object):
'''
Context Manager for unique alias generation
'''
- __slots__ = 'local'
+ __slots__ = ()
local = local()
local.alias = None
@@ -470,26 +470,29 @@ class Insert(Query):
@property
def params(self):
+ p = []
if isinstance(self.values, list):
- p = []
for values in self.values:
for value in values:
if isinstance(value, (Expression, Select)):
p.extend(value.params)
else:
p.append(value)
- return tuple(p)
elif isinstance(self.values, Select):
- return self.values.params
- else:
- return ()
+ p.extend(self.values.params)
+ if self.returning:
+ for exp in self.returning:
+ p.extend(exp.params)
+ return tuple(p)
class Update(Insert):
__slots__ = ('__where', '__values', 'from_')
- def __init__(self, table, columns, values, from_=None, where=None):
- super(Update, self).__init__(table, columns=columns, values=values)
+ def __init__(self, table, columns, values, from_=None, where=None,
+ returning=None):
+ super(Update, self).__init__(table, columns=columns, values=values,
+ returning=returning)
self.__where = None
self.from_ = From(from_) if from_ else None
self.where = where
@@ -534,7 +537,11 @@ class Update(Insert):
where = ''
if self.where:
where = ' WHERE ' + str(self.where)
- return 'UPDATE %s SET ' % table + values + from_ + where
+ returning = ''
+ if self.returning:
+ returning = ' RETURNING ' + ', '.join(map(str, self.returning))
+ return ('UPDATE %s SET ' % table + values + from_ + where
+ + returning)
@property
def params(self):
@@ -548,6 +555,9 @@ class Update(Insert):
p.extend(self.from_.params)
if self.where:
p.extend(self.where.params)
+ if self.returning:
+ for exp in self.returning:
+ p.extend(exp.params)
return tuple(p)
@@ -607,7 +617,13 @@ class Delete(Query):
@property
def params(self):
- return self.where.params if self.where else ()
+ p = []
+ if self.where:
+ p.extend(self.where.params)
+ if self.returning:
+ for exp in self.returning:
+ p.extend(exp.params)
+ return tuple(p)
class CombiningQuery(Query, FromItem, _SelectQueryMixin):
@@ -670,9 +686,9 @@ class Table(FromItem):
return Insert(self, columns=columns, values=values,
returning=returning)
- def update(self, columns, values, from_=None, where=None):
+ def update(self, columns, values, from_=None, where=None, returning=None):
return Update(self, columns=columns, values=values, from_=from_,
- where=where)
+ where=where, returning=returning)
def delete(self, only=False, using=None, where=None, returning=None):
return Delete(self, only=only, using=using, where=where,
@@ -774,7 +790,14 @@ class From(list):
return '(%s) AS "%s"' % (from_, from_.alias)
else:
alias = getattr(from_, 'alias', None)
- if alias:
+ columns_definitions = getattr(from_, 'columns_definitions',
+ None)
+ # XXX find a better test for __getattr__ which returns Column
+ if (alias and columns_definitions
+ and not isinstance(columns_definitions, Column)):
+ return '%s AS "%s" (%s)' % (from_, alias,
+ columns_definitions)
+ elif alias:
return '%s AS "%s"' % (from_, alias)
else:
return str(from_)
diff --git a/sql/functions.py b/sql/functions.py
index 58fb127..6e86138 100644
--- a/sql/functions.py
+++ b/sql/functions.py
@@ -47,13 +47,24 @@ __all__ = ['Abs', 'Cbrt', 'Ceil', 'Degrees', 'Div', 'Exp', 'Floor', 'Ln',
class Function(Expression, FromItem):
- __slots__ = ('args',)
+ __slots__ = ('args', '__columns_definitions')
table = ''
name = ''
_function = ''
- def __init__(self, *args):
+ def __init__(self, *args, **kwargs):
self.args = args
+ self.columns_definitions = kwargs.get('columns_definitions', [])
+
+ @property
+ def columns_definitions(self):
+ return ', '.join('"%s" %s' % (c, d)
+ for c, d in self.__columns_definitions)
+
+ @columns_definitions.setter
+ def columns_definitions(self, value):
+ assert isinstance(value, list)
+ self.__columns_definitions = value
@staticmethod
def _format(value):
@@ -453,7 +464,7 @@ class AtTimeZone(Function):
flavor = Flavor.get()
Mapping = flavor.function_mapping.get(self.__class__)
if Mapping:
- return str(Mapping(*self.args))
+ return str(Mapping(self.field, self.zone))
param = flavor.param
return '%s AT TIME ZONE %s' % (str(self.field), param)
diff --git a/sql/tests/test_delete.py b/sql/tests/test_delete.py
index 2ba1474..4254f44 100644
--- a/sql/tests/test_delete.py
+++ b/sql/tests/test_delete.py
@@ -52,3 +52,8 @@ class TestDelete(unittest.TestCase):
'DELETE FROM "t1" WHERE ("c" IN ('
'SELECT "a"."c" FROM "t2" AS "a"))')
self.assertEqual(query.params, ())
+
+ def test_delete_returning(self):
+ query = self.table.delete(returning=[self.table.c])
+ self.assertEqual(str(query), 'DELETE FROM "t" RETURNING "c"')
+ self.assertEqual(query.params, ())
diff --git a/sql/tests/test_functions.py b/sql/tests/test_functions.py
index 3872187..091e9b6 100644
--- a/sql/tests/test_functions.py
+++ b/sql/tests/test_functions.py
@@ -78,3 +78,18 @@ class TestFunctions(unittest.TestCase):
time_zone = AtTimeZone(self.table.c1, 'UTC')
self.assertEqual(str(time_zone), '"c1" AT TIME ZONE %s')
self.assertEqual(time_zone.params, ('UTC',))
+
+ def test_at_time_zone_mapping(self):
+ class MyAtTimeZone(Function):
+ _function = 'MY_TIMEZONE'
+
+ time_zone = AtTimeZone(self.table.c1, 'UTC')
+ flavor = Flavor(function_mapping={
+ AtTimeZone: MyAtTimeZone,
+ })
+ Flavor.set(flavor)
+ try:
+ self.assertEqual(str(time_zone), 'MY_TIMEZONE("c1", %s)')
+ self.assertEqual(time_zone.params, ('UTC',))
+ finally:
+ Flavor.set(Flavor())
diff --git a/sql/tests/test_insert.py b/sql/tests/test_insert.py
index 92954a2..283fa80 100644
--- a/sql/tests/test_insert.py
+++ b/sql/tests/test_insert.py
@@ -70,3 +70,11 @@ class TestInsert(unittest.TestCase):
self.assertEqual(str(query),
'INSERT INTO "t" ("c") VALUES (ABS(%s))')
self.assertEqual(query.params, (-1,))
+
+ def test_insert_returning(self):
+ query = self.table.insert([self.table.c1, self.table.c2],
+ [['foo', 'bar']], returning=[self.table.c1, self.table.c2])
+ self.assertEqual(str(query),
+ 'INSERT INTO "t" ("c1", "c2") VALUES (%s, %s) '
+ 'RETURNING "c1", "c2"')
+ self.assertEqual(query.params, ('foo', 'bar'))
diff --git a/sql/tests/test_select.py b/sql/tests/test_select.py
index b17142f..0adadd7 100644
--- a/sql/tests/test_select.py
+++ b/sql/tests/test_select.py
@@ -30,7 +30,7 @@
import unittest
from sql import Table, Join, Union, Literal, Flavor
-from sql.functions import Now
+from sql.functions import Now, Function
from sql.aggregate import Min
@@ -114,6 +114,17 @@ class TestSelect(unittest.TestCase):
self.assertEqual(str(query), 'SELECT * FROM NOW() AS "a"')
self.assertEqual(query.params, ())
+ def test_select_function_columns_definitions(self):
+ class Crosstab(Function):
+ _function = 'CROSSTAB'
+
+ query = Crosstab('query1', 'query2',
+ columns_definitions=[
+ ('c1', 'INT'), ('c2', 'CHAR'), ('c3', 'BOOL')]).select()
+ self.assertEqual(str(query), 'SELECT * FROM CROSSTAB(%s, %s) '
+ 'AS "a" ("c1" INT, "c2" CHAR, "c3" BOOL)')
+ self.assertEqual(query.params, ('query1', 'query2'))
+
def test_select_group_by(self):
column = self.table.c
query = self.table.select(column, group_by=column)
diff --git a/sql/tests/test_update.py b/sql/tests/test_update.py
index c0e0219..3cf1964 100644
--- a/sql/tests/test_update.py
+++ b/sql/tests/test_update.py
@@ -61,3 +61,10 @@ class TestUpdate(unittest.TestCase):
'UPDATE "t1" SET "c" = ('
'SELECT "b"."c" FROM "t2" AS "b" WHERE ("b"."i" = "t1"."i"))')
self.assertEqual(query.params, ())
+
+ def test_update_returning(self):
+ query = self.table.update([self.table.c], ['foo'],
+ returning=[self.table.c])
+ self.assertEqual(str(query),
+ 'UPDATE "t" SET "c" = %s RETURNING "t"."c"')
+ self.assertEqual(query.params, ('foo',))
--
python-sql
More information about the tryton-debian-vcs
mailing list