"""
Module to provide MongoDB functionality to Salt
:configuration: This module uses PyMongo, and accepts configuration details as
parameters as well as configuration settings::
mongodb.host: 'localhost'
mongodb.port: 27017
mongodb.user: ''
mongodb.password: ''
This data can also be passed into pillar. Options passed into opts will
overwrite options passed into pillar.
"""
import logging
import re
import salt.utils.json
from salt.exceptions import get_error_message as _get_error_message
from salt.utils.versions import Version
try:
import pymongo
HAS_MONGODB = True
except ImportError:
HAS_MONGODB = False
log = logging.getLogger(__name__)
[docs]
def __virtual__():
"""
Only load this module if pymongo is installed
"""
if HAS_MONGODB:
return "mongodb"
else:
return (
False,
"The mongodb execution module cannot be loaded: the pymongo library is not"
" available.",
)
def _connect(user=None, password=None, host=None, port=None, database="admin", authdb=None):
"""
Returns a tuple of (user, host, port) with config, pillar, or default
values assigned to missing values.
"""
if not user:
user = __salt__["config.option"]("mongodb.user")
if not password:
password = __salt__["config.option"]("mongodb.password")
if not host:
host = __salt__["config.option"]("mongodb.host")
if not port:
port = __salt__["config.option"]("mongodb.port")
if not authdb:
authdb = database
try:
conn = pymongo.MongoClient(host=host, port=port)
mdb = pymongo.database.Database(conn, database)
if user and password:
mdb.authenticate(user, password, source=authdb)
except pymongo.errors.PyMongoError:
log.error("Error connecting to database %s", database)
return False
return conn
def _to_dict(objects):
"""
Potentially interprets a string as JSON for usage with mongo
"""
try:
if isinstance(objects, str):
objects = salt.utils.json.loads(objects)
except ValueError as err:
log.error("Could not parse objects: %s", err)
raise
return objects
[docs]
def db_list(user=None, password=None, host=None, port=None, authdb=None):
"""
List all MongoDB databases
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.db_list <user> <password> <host> <port>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Listing databases")
return conn.list_database_names()
except pymongo.errors.PyMongoError as err:
log.error(err)
return str(err)
[docs]
def db_exists(name, user=None, password=None, host=None, port=None, authdb=None):
"""
Checks if a database exists in MongoDB
name
The name of the database to check for.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.db_exists <name> <user> <password> <host> <port>
"""
dbs = db_list(user, password, host, port, authdb=authdb)
if isinstance(dbs, str):
return False
return name in dbs
[docs]
def db_remove(name, user=None, password=None, host=None, port=None, authdb=None):
"""
Remove a MongoDB database
name
The name of the database to remove.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.db_remove <name> <user> <password> <host> <port>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Removing database %s", name)
conn.drop_database(name)
except pymongo.errors.PyMongoError as err:
log.error("Removing database %s failed with error: %s", name, err)
return str(err)
return True
def _version(mdb):
return mdb.command("buildInfo")["version"]
[docs]
def version(user=None, password=None, host=None, port=None, database="admin", authdb=None):
"""
Get MongoDB instance version
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.version <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
err_msg = f"Failed to connect to MongoDB database {host}:{port}"
log.error(err_msg)
return (False, err_msg)
try:
mdb = pymongo.database.Database(conn, database)
return _version(mdb)
except pymongo.errors.PyMongoError as err:
log.error("Listing users failed with error: %s", err)
return str(err)
[docs]
def user_find(name, user=None, password=None, host=None, port=None, database="admin", authdb=None):
"""
Get single user from MongoDB
name
The name of the user to find.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
database
The MongoDB database to use when looking for the user. Default is ``admin``.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.user_find <name> <user> <password> <host> <port> <database> <authdb>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
err_msg = f"Failed to connect to MongoDB database {host}:{port}"
log.error(err_msg)
return (False, err_msg)
mdb = pymongo.database.Database(conn, database)
try:
return mdb.command("usersInfo", name)["users"]
except pymongo.errors.PyMongoError as err:
log.error("Listing users failed with error: %s", err)
return (False, str(err))
[docs]
def user_list(user=None, password=None, host=None, port=None, database="admin", authdb=None):
"""
List users of a MongoDB database
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
database
The MongoDB database to use when listing users. Default is ``admin``.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.user_list <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Listing users")
mdb = pymongo.database.Database(conn, database)
output = []
mongodb_version = _version(mdb)
if Version(mongodb_version) >= Version("2.6"):
for user in mdb.command("usersInfo")["users"]:
output.append({"user": user["user"], "roles": user["roles"]})
else:
for user in mdb.system.users.find():
output.append({"user": user["user"], "readOnly": user.get("readOnly", "None")})
return output
except pymongo.errors.PyMongoError as err:
log.error("Listing users failed with error: %s", err)
return str(err)
[docs]
def user_exists(
name, user=None, password=None, host=None, port=None, database="admin", authdb=None
):
"""
Checks if a user exists in MongoDB
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
database
The MongoDB database to use when checking if the user exists. Default is ``admin``.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.user_exists <name> <user> <password> <host> <port> <database>
"""
users = user_list(user, password, host, port, database, authdb)
if isinstance(users, str):
return "Failed to connect to mongo database"
for user in users:
if name == dict(user).get("user"):
return True
return False
[docs]
def user_create(
name,
passwd,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
roles=None,
):
"""
Create a MongoDB user
name
The name of the user to create.
passwd
The password for the user that is being created.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
database
The MongoDB database to use when checking if the user exists. Default is ``admin``.
authdb
The MongoDB database to use for authentication. Default is None.
roles
The roles that should be associated with the user. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.user_create <user_name> <user_password> <roles> <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
if not roles:
roles = []
_roles = [{"role": _role, "db": database} for _role in roles]
try:
log.info("Creating user %s", name)
mdb = pymongo.database.Database(conn, database)
mdb.command("createUser", name, pwd=passwd, roles=_roles)
except pymongo.errors.PyMongoError as err:
log.error("Creating user %s failed with error: %s", name, err)
return False
return True
[docs]
def user_remove(
name, user=None, password=None, host=None, port=None, database="admin", authdb=None
):
"""
Remove a MongoDB user
name
The name of the user that should be removed.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.user_remove <name> <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Removing user %s", name)
mdb = pymongo.database.Database(conn, database)
mdb.command("dropUser", name)
except pymongo.errors.PyMongoError as err:
log.error("Removing user %s failed with error: %s", name, err)
return False
return True
[docs]
def user_roles_exists(
name, roles, database, user=None, password=None, host=None, port=None, authdb=None
):
"""
Checks if a user of a MongoDB database has specified roles
name
The name of the user to check for the specified roles.
roles
The roles to check are associated with the specified user.
database
The database to check has the specified roles for the specified user.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Examples:
.. code-block:: bash
salt '*' mongodb.user_roles_exists johndoe '["readWrite"]' dbname admin adminpwd localhost 27017
.. code-block:: bash
salt '*' mongodb.user_roles_exists johndoe '[{"role": "readWrite", "db": "dbname" }, {"role": "read", "db": "otherdb"}]' dbname admin adminpwd localhost 27017
"""
try:
roles = _to_dict(roles)
except Exception: # pylint: disable=broad-except
return "Roles provided in wrong format"
users = user_list(user, password, host, port, database, authdb)
if isinstance(users, str):
return "Failed to connect to mongo database"
for user in users:
if name == dict(user).get("user"):
for role in roles:
# if the role was provided in the shortened form, we convert it to a long form
if not isinstance(role, dict):
role = {"role": role, "db": database}
if role not in dict(user).get("roles", []):
return False
return True
return False
[docs]
def user_grant_roles(
name, roles, database, user=None, password=None, host=None, port=None, authdb=None
):
"""
Grant one or many roles to a MongoDB user
name
The user to grant the specified roles to.
roles
The roles to grant to the specified user.
database
The database to great the roles against for the specified user.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Examples:
.. code-block:: bash
salt '*' mongodb.user_grant_roles johndoe '["readWrite"]' dbname admin adminpwd localhost 27017
.. code-block:: bash
salt '*' mongodb.user_grant_roles janedoe '[{"role": "readWrite", "db": "dbname" }, {"role": "read", "db": "otherdb"}]' dbname admin adminpwd localhost 27017
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
roles = _to_dict(roles)
except Exception: # pylint: disable=broad-except
return "Roles provided in wrong format"
try:
log.info("Granting roles %s to user %s", roles, name)
mdb = pymongo.database.Database(conn, database)
mdb.command("grantRolesToUser", name, roles=roles)
except pymongo.errors.PyMongoError as err:
log.error("Granting roles %s to user %s failed with error: %s", roles, name, err)
return str(err)
return True
[docs]
def user_revoke_roles(
name, roles, database, user=None, password=None, host=None, port=None, authdb=None
):
"""
Revoke one or many roles to a MongoDB user
user
The user to connect to MongoDB as. Default is None.
roles
The roles to revoke from the specified user.
database
The database to revoke the roles from for the specified user.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Examples:
.. code-block:: bash
salt '*' mongodb.user_revoke_roles johndoe '["readWrite"]' dbname admin adminpwd localhost 27017
.. code-block:: bash
salt '*' mongodb.user_revoke_roles janedoe '[{"role": "readWrite", "db": "dbname" }, {"role": "read", "db": "otherdb"}]' dbname admin adminpwd localhost 27017
"""
conn = _connect(user, password, host, port, authdb=authdb)
if not conn:
return "Failed to connect to mongo database"
try:
roles = _to_dict(roles)
except Exception: # pylint: disable=broad-except
return "Roles provided in wrong format"
try:
log.info("Revoking roles %s from user %s", roles, name)
mdb = pymongo.database.Database(conn, database)
mdb.command("revokeRolesFromUser", name, roles=roles)
except pymongo.errors.PyMongoError as err:
log.error("Revoking roles %s from user %s failed with error: %s", roles, name, err)
return str(err)
return True
[docs]
def collection_create(
collection,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
.. versionadded:: 3006.0
Create a collection in the specified database.
collection
The name of the collection to create.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.collection_create mycollection <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Creating %s.%s", database, collection)
mdb = pymongo.database.Database(conn, database)
mdb.create_collection(collection)
except pymongo.errors.PyMongoError as err:
log.error("Creating collection %r.%r failed with error %s", database, collection, err)
return err
return True
[docs]
def collection_drop(
collection,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
.. versionadded:: 3006.0
Drop a collection in the specified database.
collection
The name of the collection to drop.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.collection_drop mycollection <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
log.info("Dropping %s.%s", database, collection)
mdb = pymongo.database.Database(conn, database)
mdb.drop_collection(collection)
except pymongo.errors.PyMongoError as err:
log.error("Creating collection %r.%r failed with error %s", database, collection, err)
return err
return True
[docs]
def collections_list(
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
.. versionadded:: 3006.0
List the collections available in the specified database.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.collections_list mycollection <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
mdb = pymongo.database.Database(conn, database)
ret = mdb.list_collection_names()
except pymongo.errors.PyMongoError as err:
log.error("Listing collections failed with error %s", err)
return err
return ret
[docs]
def insert(
objects,
collection,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
Insert an object or list of objects into a collection
objects
The objects to insert into the collection, should be provided as a list.
collection
The collection to insert the objects into.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.insert '[{"foo": "FOO", "bar": "BAR"}, {"foo": "BAZ", "bar": "BAM"}]' mycollection <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
objects = _to_dict(objects)
except Exception as err: # pylint: disable=broad-except
return err
try:
log.info("Inserting %r into %s.%s", objects, database, collection)
mdb = pymongo.database.Database(conn, database)
col = getattr(mdb, collection)
ids = col.insert_many(objects)
return ids.acknowledged
except pymongo.errors.PyMongoError as err:
log.error("Inserting objects %r failed with error %s", objects, err)
return err
[docs]
def update_one(
objects,
collection,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
Update an object into a collection
https://pymongo.readthedocs.io/en/stable/api/pymongo/collection.html#pymongo.collection.Collection.update_one
.. versionadded:: 2016.11.0
objects
The objects to update in the collection, should be provided as a list.
collection
The collection to insert the objects into.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.update_one '{"_id": "my_minion"} {"bar": "BAR"}' mycollection <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
objects = str(objects)
objs = re.split(r"}\s+{", objects)
if len(objs) != 2:
return (
"Your request does not contain a valid " + '\'{_"id": "my_id"} {"my_doc": "my_val"}\''
)
objs[0] = objs[0] + "}"
objs[1] = "{" + objs[1]
document = []
for obj in objs:
try:
obj = _to_dict(obj)
document.append(obj)
except Exception as err: # pylint: disable=broad-except
return err
_id_field = document[0]
_update_doc = document[1]
# need a string to perform the test, so using objs[0]
test_f = find(collection, objs[0], user, password, host, port, database, authdb)
if not isinstance(test_f, list):
return "The find result is not well formatted. An error appears; cannot update."
elif not test_f:
return "Did not find any result. You should try an insert before."
elif len(test_f) > 1:
return "Too many results. Please try to be more specific."
else:
try:
log.info("Updating %r into %s.%s", _id_field, database, collection)
mdb = pymongo.database.Database(conn, database)
col = getattr(mdb, collection)
ids = col.update_one(_id_field, {"$set": _update_doc})
nb_mod = ids.modified_count
return f"{nb_mod} objects updated"
except pymongo.errors.PyMongoError as err:
log.error("Updating object %s failed with error %s", objects, err)
return err
[docs]
def find(
collection,
query=None,
user=None,
password=None,
host=None,
port=None,
database="admin",
authdb=None,
):
"""
Find an object or list of objects in a collection
collection
The collection to find the objects in.
query
The query to use when locating objects in the collection.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.find mycollection '[{"foo": "FOO", "bar": "BAR"}]' <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
query = _to_dict(query)
except Exception as err: # pylint: disable=broad-except
return err
try:
log.info("Searching for %r in %s", query, collection)
mdb = pymongo.database.Database(conn, database)
col = getattr(mdb, collection)
if isinstance(query, list):
ret = []
for _query in query:
res = col.find(_query)
ret.extend(res)
else:
res = col.find(query)
ret = list(res)
return ret
except pymongo.errors.PyMongoError as err:
log.error("Searching objects failed with error: %s", err)
return err
[docs]
def remove(
collection,
query=None,
user=None,
password=None,
host=None,
port=None,
database="admin",
w=1,
authdb=None,
):
"""
Remove an object or list of objects from a collection
collection
The collection to remove objects from based on the query.
query
Query to determine which objects to remove.
user
The user to connect to MongoDB as. Default is None.
password
The password to use to connect to MongoDB as. Default is None.
host
The host where MongoDB is running. Default is None.
port
The host where MongoDB is running. Default is None.
database
The database where the collection is.
w
The number of matches to remove from the collection.
authdb
The MongoDB database to use for authentication. Default is None.
CLI Example:
.. code-block:: bash
salt '*' mongodb.remove mycollection '[{"foo": "FOO", "bar": "BAR"}, {"foo": "BAZ", "bar": "BAM"}]' <user> <password> <host> <port> <database>
"""
conn = _connect(user, password, host, port, database, authdb)
if not conn:
return "Failed to connect to mongo database"
try:
query = _to_dict(query)
except Exception as err: # pylint: disable=broad-except
return _get_error_message(err)
try:
log.info("Removing %r from %s", query, collection)
mdb = pymongo.database.Database(conn, database)
col = getattr(mdb, collection)
deleted_count = 0
if isinstance(query, list):
for _query in query:
for _ in range(0, w):
res = col.delete_one(_query)
deleted_count += res.deleted_count
else:
for _ in range(0, w):
res = col.delete_one(query)
deleted_count += res.deleted_count
return f"{deleted_count} objects removed"
except pymongo.errors.PyMongoError as err:
log.error("Removing objects failed with error: %s", _get_error_message(err))
return _get_error_message(err)