com.erudika.para.server.persistence.MongoDBDAO Maven / Gradle / Ivy
/*
* Copyright 2013-2022 Erudika. https://erudika.com
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* For issues and patches go to: https://github.com/erudika
*/
package com.erudika.para.server.persistence;
import com.erudika.para.core.App;
import com.erudika.para.core.ParaObject;
import com.erudika.para.core.annotations.Locked;
import com.erudika.para.core.persistence.DAO;
import com.erudika.para.core.utils.Config;
import com.erudika.para.core.utils.Pager;
import com.erudika.para.core.utils.Para;
import com.erudika.para.core.utils.ParaObjectUtils;
import com.erudika.para.core.utils.Utils;
import static com.erudika.para.server.persistence.MongoDBUtils.getTable;
import com.mongodb.BasicDBObject;
import com.mongodb.bulk.BulkWriteResult;
import com.mongodb.client.MongoCursor;
import com.mongodb.client.model.BulkWriteOptions;
import com.mongodb.client.model.Filters;
import com.mongodb.client.model.ReplaceOptions;
import com.mongodb.client.model.UpdateOneModel;
import com.mongodb.client.model.WriteModel;
import com.mongodb.client.result.DeleteResult;
import com.mongodb.client.result.UpdateResult;
import java.lang.annotation.Annotation;
import static java.nio.charset.StandardCharsets.UTF_8;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import org.apache.commons.lang3.StringUtils;
import org.bson.Document;
import org.bson.conversions.Bson;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* MongoDB DAO implementation for Para.
* @author Luca Venturella [[email protected]]
*/
public class MongoDBDAO implements DAO {
private static final Logger logger = LoggerFactory.getLogger(MongoDBDAO.class);
private static final String ID = "_id";
private static final String OBJECT_ID = "_ObjectId";
private static final Pattern FIELD_NAME_ENCODING_PATTERN = Pattern.compile("^Base64:.*?:(.*)$");
static {
// set up automatic table creation and deletion
App.addAppCreatedListener((App app) -> {
if (app != null && !app.isSharingTable()) {
MongoDBUtils.createTable(app.getAppIdentifier());
}
});
App.addAppDeletedListener((App app) -> {
if (app != null && !app.isSharingTable()) {
MongoDBUtils.deleteTable(app.getAppIdentifier());
}
});
}
/**
* Default constructor.
*/
public MongoDBDAO() {
}
/////////////////////////////////////////////
// CORE FUNCTIONS
/////////////////////////////////////////////
@Override
public String create(String appid, P so) {
if (so == null) {
return null;
}
if (StringUtils.isBlank(so.getId())) {
so.setId(MongoDBUtils.generateNewId());
logger.debug("Generated id: " + so.getId());
}
if (so.getTimestamp() == null) {
so.setTimestamp(Utils.timestamp());
}
so.setAppid(appid);
createRow(so.getId(), appid, toRow(so, null, false, true));
logger.debug("DAO.create() {}", so.getId());
return so.getId();
}
@Override
public
P read(String appid, String key) {
if (StringUtils.isBlank(key)) {
return null;
}
P so = fromRow(readRow(key, appid));
logger.debug("DAO.read() {} -> {}", key, so == null ? null : so.getType());
return so != null ? so : null;
}
@Override
public
void update(String appid, P so) {
if (so != null && so.getId() != null) {
so.setUpdated(Utils.timestamp());
updateRow(so.getId(), appid, toRow(so, Locked.class, true));
logger.debug("DAO.update() {}", so.getId());
}
}
@Override
public
void delete(String appid, P so) {
if (so != null && so.getId() != null) {
deleteRow(so.getId(), appid);
logger.debug("DAO.delete() {}", so.getId());
}
}
/////////////////////////////////////////////
// ROW FUNCTIONS
/////////////////////////////////////////////
private String createRow(String key, String appid, Document row) {
if (StringUtils.isBlank(key) || StringUtils.isBlank(appid) || row == null || row.isEmpty()) {
return null;
}
try {
// if there isn't a document with the same id then create a new document
// else replace the document with the same id with the new one
getTable(appid).replaceOne(new Document(ID, key), row, new ReplaceOptions().upsert(true));
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
return key;
}
//http://www.mkyong.com/mongodb/java-mongodb-update-document/
private void updateRow(String key, String appid, Document row) {
if (StringUtils.isBlank(key) || StringUtils.isBlank(appid) || row == null || row.isEmpty()) {
return;
}
try {
UpdateResult u = getTable(appid).updateOne(new Document(ID, key), new Document("$set", row));
logger.debug("key: " + key + " updated count: " + u.getModifiedCount());
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
}
private Document readRow(String key, String appid) {
if (StringUtils.isBlank(key) || StringUtils.isBlank(appid)) {
return null;
}
Document row = null;
try {
row = getTable(appid).find(new Document(ID, key)).first();
logger.debug("id: " + key + " row null: " + (row == null));
} catch (Exception e) {
logger.error(null, e);
}
return (row == null || row.isEmpty()) ? null : row;
}
private void deleteRow(String key, String appid) {
if (StringUtils.isBlank(key) || StringUtils.isBlank(appid)) {
return;
}
try {
DeleteResult d = getTable(appid).deleteOne(new Document(ID, key));
logger.debug("key: " + key + " deleted count: " + d.getDeletedCount());
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
}
/////////////////////////////////////////////
// READ ALL FUNCTIONS
/////////////////////////////////////////////
@Override
public
void createAll(String appid, List
objects) {
if (objects == null || objects.isEmpty() || StringUtils.isBlank(appid)) {
return;
}
try {
List documents = new ArrayList();
for (ParaObject so : objects) {
if (so != null) {
if (StringUtils.isBlank(so.getId())) {
so.setId(MongoDBUtils.generateNewId());
logger.debug("Generated id: " + so.getId());
}
if (so.getTimestamp() == null) {
so.setTimestamp(Utils.timestamp());
}
so.setAppid(appid);
documents.add(toRow(so, null, false, true));
}
}
if (!documents.isEmpty()) {
getTable(appid).insertMany(documents);
}
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
logger.debug("DAO.createAll() {}", objects.size());
}
@Override
public Map readAll(String appid, List keys, boolean getAllColumns) {
if (keys == null || keys.isEmpty() || StringUtils.isBlank(appid)) {
return new LinkedHashMap();
}
Map results = new LinkedHashMap(keys.size(), 0.75f, true);
BasicDBObject inQuery = new BasicDBObject();
inQuery.put(ID, new BasicDBObject("$in", keys));
MongoCursor cursor = getTable(appid).find(inQuery).iterator();
while (cursor.hasNext()) {
Document d = cursor.next();
P obj = fromRow(d);
if (d != null) {
results.put(d.getString(ID), obj);
}
}
logger.debug("DAO.readAll() {}", results.size());
return results;
}
@Override
public List
readPage(String appid, Pager pager) {
LinkedList
results = new LinkedList
();
if (StringUtils.isBlank(appid)) {
return results;
}
if (pager == null) {
pager = new Pager();
}
try {
String lastKey = pager.getLastKey();
MongoCursor cursor;
Bson filter = Filters.gt(OBJECT_ID, lastKey);
if (lastKey == null) {
cursor = getTable(appid).find().batchSize(pager.getLimit()).limit(pager.getLimit()).iterator();
} else {
cursor = getTable(appid).find(filter).batchSize(pager.getLimit()).limit(pager.getLimit()).iterator();
}
while (cursor.hasNext()) {
Map row = documentToMap(cursor.next());
P obj = fromRow(row);
if (obj != null) {
results.add(obj);
pager.setLastKey((String) row.get(OBJECT_ID));
}
}
if (!results.isEmpty()) {
pager.setCount(pager.getCount() + results.size());
}
} catch (Exception e) {
logger.error(null, e);
}
logger.debug("readPage() page: {}, results:", pager.getPage(), results.size());
return results;
}
@Override
public void updateAll(String appid, List
objects) {
if (StringUtils.isBlank(appid) || objects == null) {
return;
}
try {
ArrayList> updates = new ArrayList>();
List ids = new ArrayList(objects.size());
for (P object : objects) {
if (object != null) {
object.setUpdated(Utils.timestamp());
Document id = new Document(ID, object.getId());
Document data = new Document("$set", toRow(object, Locked.class, true));
UpdateOneModel um = new UpdateOneModel(id, data);
updates.add(um);
ids.add(object.getId());
}
}
BulkWriteResult res = getTable(appid).bulkWrite(updates, new BulkWriteOptions().ordered(true));
logger.debug("Updated: " + res.getModifiedCount() + ", keys: " + ids);
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
logger.debug("DAO.updateAll() {}", objects.size());
}
@Override
public void deleteAll(String appid, List
objects) {
if (objects == null || objects.isEmpty() || StringUtils.isBlank(appid)) {
return;
}
try {
BasicDBObject query = new BasicDBObject();
List list = new ArrayList();
for (ParaObject object : objects) {
list.add(object.getId());
}
query.put(ID, new BasicDBObject("$in", list));
getTable(appid).deleteMany(query);
logger.debug("DAO.deleteAll() {}", objects.size());
} catch (Exception e) {
logger.error(null, e);
throwIfNecessary(e);
}
}
/////////////////////////////////////////////
// MISC FUNCTIONS
/////////////////////////////////////////////
private Document toRow(P so, Class extends Annotation> filter, boolean setNullFields) {
return toRow(so, filter, setNullFields, false);
}
@SuppressWarnings("unchecked")
private
Document toRow(P so, Class extends Annotation> filter,
boolean setNullFields, boolean setMongoId) {
Document row = new Document();
if (so == null) {
return row;
}
// field values will be stored as they are - object structure and types will be preserved
for (Entry entry : ParaObjectUtils.getAnnotatedFields(so, filter, false).entrySet()) {
Object value = entry.getValue();
if (value != null && (!StringUtils.isBlank(value.toString()) || setNullFields)) {
// "id" in ParaObject is translated to "_ID" mongodb
if (entry.getKey().equals(Config._ID)) {
row.put(ID, value.toString());
} else {
if (value instanceof Map) {
row.put(sanitizeField(entry.getKey()), sanitizeFields((Map) value));
} else {
row.put(sanitizeField(entry.getKey()), value);
}
}
if (setMongoId) {
// we add the native MongoDB id which will later be used for pagination and sorting
row.put(OBJECT_ID, MongoDBUtils.generateNewId());
}
}
}
return row;
}
private P fromRow(Document row) {
return fromRow(documentToMap(row));
}
private
P fromRow(Map row) {
return ParaObjectUtils.setAnnotatedFields(row);
}
@SuppressWarnings("unchecked")
private Map documentToMap(Document row) {
if (row == null || row.isEmpty()) {
logger.debug("row is null or empty");
return Collections.emptyMap();
}
Map props = new HashMap();
for (Entry col : row.entrySet()) {
Object value = col.getValue();
// "_ID" mongodb is translated to "id" in ParaObject
if (col.getKey().equals(ID)) {
props.put(Config._ID, value);
} else {
if (value instanceof Map) {
props.put(desanitizeField(col.getKey()), desanitizeFields((Map) value));
} else {
props.put(desanitizeField(col.getKey()), value);
}
}
}
return props;
}
private static void throwIfNecessary(Throwable t) {
if (t != null && Para.getConfig().exceptionOnWriteErrorsEnabled()) {
throw new RuntimeException("DAO write operation failed!", t);
}
}
/**
* MongoDB doesn't like '$' and '.' in field names. This replaces all '.' and the first '$'
* with '{Base64(.|$)}'. Ref: https://github.com/Erudika/scoold/issues/11
* @param fieldName the old document key
* @return a sanitized key
*/
String sanitizeField(String fieldName) {
if (!StringUtils.contains(fieldName, ".") && !StringUtils.startsWith(fieldName, "$")) {
return fieldName;
}
String b = "Base64:" + fieldName.replaceAll("^\\$+", "").replaceAll("\\.", "_") + ":" +
Utils.base64enc(fieldName.getBytes(UTF_8));
return b;
}
String desanitizeField(String fieldName) {
if (fieldName != null) {
Matcher m = FIELD_NAME_ENCODING_PATTERN.matcher(fieldName);
if (m.matches()) {
return Utils.base64dec(m.group(1));
}
}
return fieldName;
}
@SuppressWarnings("unchecked")
Map sanitizeFields(Map row) {
if (row != null) {
Map cleanRow = new HashMap(row.size());
for (Entry entry : row.entrySet()) {
String key = entry.getKey();
Object value = entry.getValue();
if (value instanceof Map) {
cleanRow.put(sanitizeField(key), sanitizeFields((Map) value));
} else {
cleanRow.put(sanitizeField(key), value);
}
}
return cleanRow;
}
return null;
}
@SuppressWarnings("unchecked")
Map desanitizeFields(Map row) {
if (row != null) {
Map cleanRow = new HashMap(row.size());
for (Entry entry : row.entrySet()) {
String key = entry.getKey();
Object value = entry.getValue();
if (value instanceof Map) {
cleanRow.put(desanitizeField(key), desanitizeFields((Map) value));
} else {
cleanRow.put(desanitizeField(key), value);
}
}
return cleanRow;
}
return null;
}
//////////////////////////////////////////////////////
@Override
public String create(P so) {
return create(Para.getConfig().getRootAppIdentifier(), so);
}
@Override
public
P read(String key) {
return read(Para.getConfig().getRootAppIdentifier(), key);
}
@Override
public
void update(P so) {
update(Para.getConfig().getRootAppIdentifier(), so);
}
@Override
public
void delete(P so) {
delete(Para.getConfig().getRootAppIdentifier(), so);
}
@Override
public
void createAll(List
objects) {
createAll(Para.getConfig().getRootAppIdentifier(), objects);
}
@Override
public
Map readAll(List keys, boolean getAllColumns) {
return readAll(Para.getConfig().getRootAppIdentifier(), keys, getAllColumns);
}
@Override
public List
readPage(Pager pager) {
return readPage(Para.getConfig().getRootAppIdentifier(), pager);
}
@Override
public
void updateAll(List
objects) {
updateAll(Para.getConfig().getRootAppIdentifier(), objects);
}
@Override
public
void deleteAll(List
objects) {
deleteAll(Para.getConfig().getRootAppIdentifier(), objects);
}
}