-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathSqlCreateRule.h
143 lines (124 loc) · 5.21 KB
/
SqlCreateRule.h
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
/*
Copyright (C) 2011-2013 Klarälvdalens Datakonsult AB,
a KDAB Group company, info@kdab.net,
author Volker Krause <volker.krause@kdab.com>
This library is free software; you can redistribute it and/or modify it
under the terms of the GNU Library General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at your
option) any later version.
This library is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public
License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to the
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301, USA.
*/
#ifndef SQLCREATERULE_H
#define SQLCREATERULE_H
#include "SqlInternals_p.h"
#include "SqlSchema_p.h"
#include "SqlQuery.h"
#include "SqlUtils.h"
#include <boost/mpl/for_each.hpp>
#include <boost/mpl/placeholders.hpp>
#include <QStringList>
/**
* @file SqlCreateRule.h
* Classes and functions to create tables rules based on the schema definition in SqlSchema.h
*/
namespace Sql {
namespace detail {
/**
* Table-wise notification rule creator.
* @internal
*/
struct notification_rule_creator {
notification_rule_creator( QStringList &stmts ) : m_stmts( stmts ) {}
template <typename T>
void operator()( wrap<T> ) {
if ( T::is_lookup_table::value || T::is_relation::value )
return;
const QStringList ops = QStringList() << QLatin1String( "Insert" ) << QLatin1String( "Update" ) << QLatin1String( "Delete" );
foreach ( const QString& op, ops ) {
const QString stmt = QLatin1Literal( "CREATE OR REPLACE RULE " )
% T::tableName()
% QLatin1Literal( "Notification" ) % op % QLatin1Literal( "Rule AS ON " )
% op.toUpper() % QLatin1Literal( " TO " )
% T::tableName()
% QLatin1Literal( " DO ALSO NOTIFY " )
% T::tableName()
% QLatin1Literal( "Changed" );
m_stmts.push_back( stmt );
}
}
QStringList &m_stmts;
};
/**
* Column-wise notification rule creator.
* @internal
*/
struct column_notification_rule_creator {
column_notification_rule_creator( QStringList &stmts ) : m_stmts( stmts ) {}
template <typename T>
void operator()( wrap<T> ) {
if( T::notify::value ) //notify with the corresponding column content ( as the notify channel) when a row is modified
{
//be sure to have unique rule name and stay within the allowed name size. We don't care that it's cryptic as it isn't used anywhere. so use Uuids.
QString ruleID = QUuid::createUuid().toString();
ruleID.replace(QLatin1Char('{'), QLatin1Char('\"'));
ruleID.replace(QLatin1Char('}'), QLatin1Char('\"'));
const QString identifier = SqlUtils::createIdentifier(T::table::sqlName() % QLatin1Literal( "_" ) % T::sqlName());
const QString stmt = QLatin1Literal( "CREATE OR REPLACE RULE " )
% ruleID % QLatin1Literal( " AS ON UPDATE TO " )
% T::table::sqlName()
% QLatin1Literal( " DO ALSO SELECT pg_notify(CAST (OLD." ) % T::sqlName()
% QLatin1Literal( " AS text) || '_") % identifier % QLatin1Literal( "','')");
m_stmts.push_back( stmt );
}
}
QStringList &m_stmts;
};
/**
* Table-wise notification rule creator (create statements for each Notify-enabled column in all tables).
* @internal
*/
struct table_notification_rule_creator {
table_notification_rule_creator( QStringList &stmts ) : m_stmts( stmts ) {}
template <typename T>
void operator()( wrap<T> ) {
boost::mpl::for_each<typename T::columns, detail::wrap<boost::mpl::placeholders::_1> >( column_notification_rule_creator( m_stmts ) );
}
QStringList &m_stmts;
};
}
/**
* Returns a list of CREATE RULE statements to create notification rules for all mutable tables.
* @tparam Schema A MPL sequence of tables.
* @returns A list of SQL statements to create notification rules for all mutable tables.
*/
template<typename Schema>
QStringList createNotificationRuleStatements()
{
QStringList statements;
boost::mpl::for_each<Schema, detail::wrap<boost::mpl::placeholders::_1> >( detail::notification_rule_creator( statements ) );
boost::mpl::for_each<Schema, detail::wrap<boost::mpl::placeholders::_1> >( detail::table_notification_rule_creator( statements ) );
return statements;
}
/**
* Create all table change notification rules.
* @tparam Schema A MPL sequence of tables.
* @param db The database to create the tables in
* @throws SqlException in case of a database error
*/
template <typename Schema>
void createNotificationRules( const QSqlDatabase &db )
{
foreach ( const QString &stmt, Sql::createNotificationRuleStatements<Schema>() ) {
SqlQuery q( db );
q.exec( stmt );
}
}
}
#endif