123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501 |
- /*
- * Copyright (c) 2015, 2018, Oracle and/or its affiliates. All rights reserved.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License, version 2.0, as
- * published by the Free Software Foundation.
- *
- * This program is also distributed with certain software (including
- * but not limited to OpenSSL) that is licensed under separate terms,
- * as designated in a particular file or component or in included license
- * documentation. The authors of MySQL hereby grant you an
- * additional permission to link the program and your derivative works
- * with the separately licensed software that they have included with
- * MySQL.
- *
- * Without limiting anything contained in the foregoing, this file,
- * which is part of MySQL Connector/C++, is also subject to the
- * Universal FOSS Exception, version 1.0, a copy of which can be found at
- * http://oss.oracle.com/licenses/universal-foss-exception.
- *
- * This program 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 General Public License, version 2.0, for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software Foundation, Inc.,
- * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
- */
- #ifndef MYSQLX_CRUD_H
- #define MYSQLX_CRUD_H
- /**
- @file
- Common templates used to define CRUD operation classes.
- */
- #include "common.h"
- #include "detail/crud.h"
- namespace mysqlx {
- class Collection;
- class Table;
- namespace internal {
- /*
- Factory for constructing concrete implementations of various CRUD
- operations. All these implementations implement the base Executable_if
- interface.
- Note: The caller of mk_xxx() method takes ownership of the returned
- implementation object.
- */
- struct PUBLIC_API Crud_factory
- {
- using Impl = common::Executable_if;
- static Impl* mk_add(Collection &coll);
- static Impl* mk_remove(Collection &coll, const string &expr);
- static Impl* mk_find(Collection &coll);
- static Impl* mk_find(Collection &coll, const string &expr);
- static Impl* mk_modify(Collection &coll, const string &expr);
- static Impl* mk_insert(Table &tbl);
- static Impl* mk_select(Table &tbl);
- static Impl* mk_update(Table &tbl);
- static Impl* mk_remove(Table &tbl);
- static Impl* mk_sql(Session &sess, const string &sql);
- };
- } // internal
- } // mysqlx
- /*
- Different CRUD operation classes derive from `Executable` which defines
- the `execute()` method that executes given operation. Derived classes
- define additional methods that can modify the operation before it gets
- executed.
- The hierarchy of classes reflects the grammar that defines the order in which
- fluent API calls can be done. It is built using templates, such as Offset<>
- below, which add one API call on top of base class which defines remaining
- API calls that can be called later. For example, type
- Limit< Offset< Executable<...> > >
- represents an operation for which first .limit() can be called, followed by
- .offset() and then finally .execute(). See classes like
- Collection_find_base in collection_crud.h for more examples.
- Each template assumes that its base class defines method 'get_impl()' which
- returns a pointer to the internal implementation object. It also assumes that
- this implementation is of appropriate type and can be casted to
- the appropriate interface type. For example Limit<> template assumes
- that the implementation type can be casted to Limit_if type.
- */
- namespace mysqlx {
- /**
- @brief The LockContention enum defines constants for defining the row locking contention for `Set_lock::lockExclusive()` and `Set_lock::lockShared()` methods.
- @see https://dev.mysql.com/doc/refman/8.0/en/innodb-locking-reads.html#innodb-locking-reads-nowait-skip-locked
- */
- enum_class LockContention
- {
- #define DEVAPI_LOCK_CONTENTION_ENUM(X,N) X = N,
- LOCK_CONTENTION_LIST(DEVAPI_LOCK_CONTENTION_ENUM)
- };
- namespace internal {
- /**
- Template for defining fluent api for CRUD operations.
- */
- template <class Base>
- class Offset
- : public Base
- {
- using Operation = Base;
- public:
- /**
- Skip the given number of items (rows or documents) before starting
- to perform the operation.
- */
- Operation& offset(unsigned rows)
- {
- try {
- get_impl()->set_offset(rows);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Limit_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Limit
- : public Base
- {
- using Operation = Base;
- public:
- /**
- %Limit the operation to the given number of items (rows or documents).
- */
- Operation& limit(unsigned items)
- {
- try {
- get_impl()->set_limit(items);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Limit_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Sort
- : public Base
- , Sort_detail
- {
- using Operation = Base;
- public:
- /**
- Specify ordering of documents in a query results.
- Arguments are one or more strings of the form `"<expr> <dir>"` where
- `<expr>` gives the value to sort on and `<dir>` is a sorting direction
- `ASC` or `DESC`.
- */
- template <typename...Type>
- Operation& sort(Type... spec)
- {
- try {
- add_sort(get_impl(), spec...);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Sort_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Order_by
- : public Base
- , Sort_detail
- {
- using Operation = Base;
- public:
- /**
- Specify ordering of rows in a query results.
- Arguments are one or more strings of the form `"<expr> <dir>"` where
- `<expr>` gives the value to sort on and `<dir>` is a sorting direction
- `ASC` or `DESC`.
- */
- template <typename...Type>
- Operation& orderBy(Type... spec)
- {
- try {
- add_sort(get_impl(), spec...);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Sort_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Having
- : public Base
- {
- using Operation = Base;
- public:
- /**
- Specify filter over grouped results of a query.
- The argument is a Boolean expression which can use aggregation functions.
- */
- Operation& having(const string& having_spec)
- {
- try {
- get_impl()->set_having(having_spec);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Having_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Group_by
- : public Base
- , Group_by_detail
- {
- using Operation = Base;
- public:
- /**
- Specify grouping of items in a query result.
- Arguments are a one or more expressions. Documents/rows for which
- expressions evaluate to the same value are grouped together.
- */
- template <typename... Expr>
- Operation& groupBy(Expr... group_by_spec)
- {
- try {
- do_group_by(get_impl(), group_by_spec...);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Group_by_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Bind_placeholders
- : public Base
- , Bind_detail
- {
- using BindOperation = Bind_placeholders;
- public:
- /**
- Specify values for '?' placeholders in a query.
- One or more values can be specified in a single call to bind(). A query
- can be executed only if values for all placeholders have been specified.
- */
- template <typename... Types>
- BindOperation& bind(Types&&... vals)
- {
- try {
- add_params(get_impl(), std::forward<Types>(vals)...);
- return *this;
- }
- CATCH_AND_WRAP
- }
- protected:
- using Impl = common::Bind_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base>
- class Bind_parameters
- : public Base
- {
- using BindOperation = Bind_parameters;
- using Operation = Base;
- public:
- /**
- Bind parameter with given name to the given value.
- A statement or query can be executed only if all named parameters used by
- it are bound to values.
- */
- BindOperation& bind(const string ¶meter, const Value &val)
- {
- if (Value::DOCUMENT == val.getType())
- throw_error("Can not bind a parameter to a document");
- if (Value::ARRAY == val.getType())
- throw_error("Can not bind a parameter to an array");
- try {
- get_impl()->add_param(parameter, (const common::Value&)val);
- return *this;
- }
- CATCH_AND_WRAP
- }
- /**
- Bind parameters to values given by a map from parameter
- names to their values.
- */
- template <class Map>
- Operation& bind(const Map &args)
- {
- for (const auto &keyval : args)
- {
- bind(keyval.first, keyval.second);
- }
- return *this;
- }
- protected:
- using Impl = common::Bind_if;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- /// @copydoc Offset
- template <class Base, class IMPL>
- class Set_lock
- : public Base
- {
- using Operation = Base;
- public:
- /**
- Set a shared mode lock on any rows/documents that are read.
- Other sessions can read, but not modify locked rows/documents.
- */
- Operation&
- lockShared(LockContention contention= LockContention::DEFAULT)
- {
- get_impl()->set_lock_mode(common::Lock_mode::SHARED,
- common::Lock_contention((unsigned)contention));
- return *this;
- }
- /**
- Set an exclusive mode lock on any rows/documents that are read.
- Other sessions are blocked from modifying, locking, or reading the data
- in certain transaction isolation levels. The lock is released
- when the transaction is committed or rolled back.
- */
- Operation&
- lockExclusive(LockContention contention = LockContention::DEFAULT)
- {
- get_impl()->set_lock_mode(common::Lock_mode::EXCLUSIVE,
- common::Lock_contention((unsigned)contention));
- return *this;
- }
- protected:
- using Impl = IMPL;
- Impl* get_impl()
- {
- return static_cast<Impl*>(Base::get_impl());
- }
- };
- }} // mysqlx::internal
- #endif
|