/* * Copyright 2012 - 2022 Anton Tananaev (anton@traccar.org) * * 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. */ package org.traccar.database; import com.zaxxer.hikari.HikariConfig; import com.zaxxer.hikari.HikariDataSource; import liquibase.Contexts; import liquibase.Liquibase; import liquibase.database.Database; import liquibase.database.DatabaseFactory; import liquibase.exception.LiquibaseException; import liquibase.resource.FileSystemResourceAccessor; import liquibase.resource.ResourceAccessor; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.traccar.Context; import org.traccar.config.Config; import org.traccar.config.Keys; import org.traccar.model.Attribute; import org.traccar.model.BaseModel; import org.traccar.model.Calendar; import org.traccar.model.Command; import org.traccar.model.Device; import org.traccar.model.Driver; import org.traccar.model.Event; import org.traccar.model.Geofence; import org.traccar.model.Group; import org.traccar.model.Maintenance; import org.traccar.model.ManagedUser; import org.traccar.model.Notification; import org.traccar.model.Permission; import org.traccar.model.Position; import org.traccar.model.Server; import org.traccar.model.Statistics; import org.traccar.model.User; import org.traccar.storage.DatabaseStorage; import org.traccar.storage.QueryBuilder; import org.traccar.storage.Storage; import org.traccar.storage.StorageException; import org.traccar.storage.query.Columns; import org.traccar.storage.query.Condition; import org.traccar.storage.query.Limit; import org.traccar.storage.query.Order; import org.traccar.storage.query.Request; import javax.sql.DataSource; import java.beans.Introspector; import java.io.File; import java.lang.reflect.Method; import java.net.URL; import java.sql.SQLException; import java.util.Collection; import java.util.Date; import java.util.LinkedList; import java.util.List; public class DataManager { private static final Logger LOGGER = LoggerFactory.getLogger(DataManager.class); public static final String ACTION_SELECT_ALL = "selectAll"; public static final String ACTION_SELECT = "select"; public static final String ACTION_INSERT = "insert"; public static final String ACTION_UPDATE = "update"; public static final String ACTION_DELETE = "delete"; private final Config config; private DataSource dataSource; public DataSource getDataSource() { return dataSource; } private final Storage storage; private boolean generateQueries; private final boolean forceLdap; public DataManager(Config config) throws Exception { this.config = config; forceLdap = config.getBoolean(Keys.LDAP_FORCE); initDatabase(); initDatabaseSchema(); storage = new DatabaseStorage(dataSource); } private void initDatabase() throws Exception { String driverFile = config.getString(Keys.DATABASE_DRIVER_FILE); if (driverFile != null) { ClassLoader classLoader = ClassLoader.getSystemClassLoader(); try { Method method = classLoader.getClass().getDeclaredMethod("addURL", URL.class); method.setAccessible(true); method.invoke(classLoader, new File(driverFile).toURI().toURL()); } catch (NoSuchMethodException e) { Method method = classLoader.getClass() .getDeclaredMethod("appendToClassPathForInstrumentation", String.class); method.setAccessible(true); method.invoke(classLoader, driverFile); } } String driver = config.getString(Keys.DATABASE_DRIVER); if (driver != null) { Class.forName(driver); } HikariConfig hikariConfig = new HikariConfig(); hikariConfig.setDriverClassName(driver); hikariConfig.setJdbcUrl(config.getString(Keys.DATABASE_URL)); hikariConfig.setUsername(config.getString(Keys.DATABASE_USER)); hikariConfig.setPassword(config.getString(Keys.DATABASE_PASSWORD)); hikariConfig.setConnectionInitSql(config.getString(Keys.DATABASE_CHECK_CONNECTION)); hikariConfig.setIdleTimeout(600000); int maxPoolSize = config.getInteger(Keys.DATABASE_MAX_POOL_SIZE); if (maxPoolSize != 0) { hikariConfig.setMaximumPoolSize(maxPoolSize); } generateQueries = config.getBoolean(Keys.DATABASE_GENERATE_QUERIES); dataSource = new HikariDataSource(hikariConfig); } public static String constructPermissionQuery(String action, Class owner, Class property) { switch (action) { case ACTION_SELECT_ALL: return "SELECT " + makeNameId(owner) + ", " + makeNameId(property) + " FROM " + getPermissionsTableName(owner, property); case ACTION_INSERT: return "INSERT INTO " + getPermissionsTableName(owner, property) + " (" + makeNameId(owner) + ", " + makeNameId(property) + ") VALUES (:" + makeNameId(owner) + ", :" + makeNameId(property) + ")"; case ACTION_DELETE: return "DELETE FROM " + getPermissionsTableName(owner, property) + " WHERE " + makeNameId(owner) + " = :" + makeNameId(owner) + " AND " + makeNameId(property) + " = :" + makeNameId(property); default: throw new IllegalArgumentException("Unknown action"); } } public String getQuery(String action, Class owner, Class property) { String queryName; switch (action) { case ACTION_SELECT_ALL: queryName = "database.select" + owner.getSimpleName() + property.getSimpleName() + "s"; break; case ACTION_INSERT: queryName = "database.link" + owner.getSimpleName() + property.getSimpleName(); break; default: queryName = "database.unlink" + owner.getSimpleName() + property.getSimpleName(); break; } String query = config.getString(queryName); if (query == null) { if (generateQueries) { query = constructPermissionQuery( action, owner, property.equals(User.class) ? ManagedUser.class : property); } else { LOGGER.info("Query not provided: " + queryName); } } return query; } private static String getPermissionsTableName(Class owner, Class property) { String propertyName = property.getSimpleName(); if (propertyName.equals("ManagedUser")) { propertyName = "User"; } return "tc_" + Introspector.decapitalize(owner.getSimpleName()) + "_" + Introspector.decapitalize(propertyName); } private static String getObjectsTableName(Class clazz) { String result = "tc_" + Introspector.decapitalize(clazz.getSimpleName()); // Add "s" ending if object name is not plural already if (!result.endsWith("s")) { result += "s"; } return result; } private void initDatabaseSchema() throws LiquibaseException { if (config.hasKey(Keys.DATABASE_CHANGELOG)) { ResourceAccessor resourceAccessor = new FileSystemResourceAccessor(new File(".")); Database database = DatabaseFactory.getInstance().openDatabase( config.getString(Keys.DATABASE_URL), config.getString(Keys.DATABASE_USER), config.getString(Keys.DATABASE_PASSWORD), config.getString(Keys.DATABASE_DRIVER), null, null, null, resourceAccessor); Liquibase liquibase = new Liquibase( config.getString(Keys.DATABASE_CHANGELOG), resourceAccessor, database); liquibase.clearCheckSums(); liquibase.update(new Contexts()); } } public User login(String email, String password) throws StorageException { User user = storage.getObject(User.class, new Request( new Columns.Include("id", "login", "hashedPassword", "salt"), new Condition.Or( new Condition.Equals("email", "email", email.trim()), new Condition.Equals("login", "email")))); LdapProvider ldapProvider = Context.getLdapProvider(); if (user != null) { if (ldapProvider != null && user.getLogin() != null && ldapProvider.login(user.getLogin(), password) || !forceLdap && user.isPasswordValid(password)) { return user; } } else { if (ldapProvider != null && ldapProvider.login(email, password)) { user = ldapProvider.getUser(email); Context.getUsersManager().addItem(user); return user; } } return null; } public void updateUserPassword(User user) throws StorageException { storage.updateObject(user, new Request( new Columns.Include("hashedPassword", "salt"), new Condition.Equals("id", "id"))); } public void updateDeviceStatus(Device device) throws StorageException { storage.updateObject(device, new Request( new Columns.Include("lastUpdate"), new Condition.Equals("id", "id"))); } public Collection getPositions(long deviceId, Date from, Date to) throws StorageException { return storage.getObjects(Position.class, new Request( new Columns.All(), new Condition.And( new Condition.Equals("deviceId", "deviceId", deviceId), new Condition.Between("fixTime", "from", from, "to", to)), new Order("fixTime"))); } public Position getPrecedingPosition(long deviceId, Date date) throws StorageException { return storage.getObject(Position.class, new Request( new Columns.All(), new Condition.And( new Condition.Equals("deviceId", "deviceId", deviceId), new Condition.Compare("fixTime", "<=", "time", date)), new Order(true, "fixTime"), new Limit(1))); } public void updateLatestPosition(Position position) throws StorageException { Device device = new Device(); device.setId(position.getDeviceId()); device.setPositionId(position.getId()); storage.updateObject(device, new Request( new Columns.Include("positionId"), new Condition.Equals("id", "id"))); } public Collection getLatestPositions() throws StorageException { List positions = new LinkedList<>(); List devices = storage.getObjects(Device.class, new Request(new Columns.Include("positionId"))); for (Device device : devices) { positions.addAll(storage.getObjects(Position.class, new Request( new Columns.All(), new Condition.Equals("id", "id", device.getPositionId())))); } return positions; } public Server getServer() throws StorageException { return storage.getObject(Server.class, new Request(new Columns.All())); } public Collection getEvents(long deviceId, Date from, Date to) throws StorageException { return storage.getObjects(Event.class, new Request( new Columns.All(), new Condition.And( new Condition.Equals("deviceId", "deviceId", deviceId), new Condition.Between("eventTime", "from", from, "to", to)), new Order("eventTime"))); } public Collection getStatistics(Date from, Date to) throws StorageException { return storage.getObjects(Statistics.class, new Request( new Columns.All(), new Condition.Between("captureTime", "from", from, "to", to), new Order("captureTime"))); } public static Class getClassByName(String name) throws ClassNotFoundException { switch (name.toLowerCase().replace("id", "")) { case "device": return Device.class; case "group": return Group.class; case "user": return User.class; case "manageduser": return ManagedUser.class; case "geofence": return Geofence.class; case "driver": return Driver.class; case "attribute": return Attribute.class; case "calendar": return Calendar.class; case "command": return Command.class; case "maintenance": return Maintenance.class; case "notification": return Notification.class; case "order": return org.traccar.model.Order.class; default: throw new ClassNotFoundException(); } } private static String makeNameId(Class clazz) { String name = clazz.getSimpleName(); return Introspector.decapitalize(name) + (!name.contains("Id") ? "Id" : ""); } public Collection getPermissions(Class owner, Class property) throws StorageException, ClassNotFoundException { try { return QueryBuilder.create(dataSource, getQuery(ACTION_SELECT_ALL, owner, property)) .executePermissionsQuery(); } catch (SQLException e) { throw new StorageException(e); } } public void linkObject(Class owner, long ownerId, Class property, long propertyId, boolean link) throws StorageException { try { QueryBuilder.create(dataSource, getQuery(link ? ACTION_INSERT : ACTION_DELETE, owner, property)) .setLong(makeNameId(owner), ownerId) .setLong(makeNameId(property), propertyId) .executeUpdate(); } catch (SQLException e) { throw new StorageException(e); } } public T getObject(Class clazz, long entityId) throws StorageException { return storage.getObject(clazz, new Request( new Columns.All(), new Condition.Equals("id", "id", entityId))); } public Collection getObjects(Class clazz) throws StorageException { return storage.getObjects(clazz, new Request(new Columns.All())); } public void addObject(BaseModel entity) throws StorageException { entity.setId(storage.addObject(entity, new Request(new Columns.Exclude("id")))); } public void updateObject(BaseModel entity) throws StorageException { storage.updateObject(entity, new Request( new Columns.Exclude("id"), new Condition.Equals("id", "id"))); } public void removeObject(Class clazz, long entityId) throws StorageException { storage.removeObject(clazz, new Request(new Condition.Equals("id", "id", entityId))); } }