All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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.ReplaceOneModel;
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 { // fix duplicate _id errors by using a map List> bulkOperations = 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); Document doc = toRow(so, null, false, true); Object id = doc.get(ID); doc.remove(ID); // fix MongoWriteConcernException error bulkOperations.add(new ReplaceOneModel<>(Filters.eq(ID, id), doc, new ReplaceOptions().upsert(true))); } } if (!bulkOperations.isEmpty()) { getTable(appid).bulkWrite(bulkOperations); } } 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 filter, boolean setNullFields) { return toRow(so, filter, setNullFields, false); } @SuppressWarnings("unchecked") private

Document toRow(P so, Class 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); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy