/*
 *    SLDEditor - SLD Editor application
 *
 *    (C) 2016, SCISYS
 *
 */

package com.sldeditor.tool.dbconnectionlist;

import com.sldeditor.common.data.DatabaseConnection;
import com.sldeditor.common.data.DatabaseConnectionField;
import com.sldeditor.common.localisation.Localisation;
import com.sldeditor.datasource.extension.filesystem.node.file.FileHandlerInterface;
import com.sldeditor.extension.filesystem.file.database.DatabaseFileHandler;
import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import javax.swing.filechooser.FileNameExtensionFilter;
import org.geotools.data.DataAccessFactory.Param;
import org.geotools.data.DataStoreFactorySpi;
import org.geotools.data.DataStoreFinder;
import org.geotools.data.db2.DB2NGDataStoreFactory;
import org.geotools.data.h2.H2DataStoreFactory;
import org.geotools.data.mysql.MySQLDataStoreFactory;
import org.geotools.data.oracle.OracleNGDataStoreFactory;
import org.geotools.data.postgis.PostgisNGDataStoreFactory;
import org.geotools.data.spatialite.SpatiaLiteDataStoreFactory;
import org.geotools.data.sqlserver.SQLServerDataStoreFactory;
import org.geotools.data.sqlserver.jtds.JDTSSQLServerJNDIDataStoreFactory;
import org.geotools.data.sqlserver.jtds.JTDSSqlServerDataStoreFactory;
import org.geotools.data.teradata.TeradataDataStoreFactory;
import org.geotools.geopkg.GeoPkgDataStoreFactory;
import org.geotools.jdbc.JDBCDataStoreFactory;

/**
 * A factory for creating DatabaseConnection objects.
 *
 * @author Robert Ward (SCISYS)
 */
public class DatabaseConnectionFactory {

    /** The Constant CONNECTION_STRING_4. */
    private static final String CONNECTION_STRING_4 = "%s/%[email protected]%s:%s";

    /** The Constant GEOPACKAGE_FILE_EXTENSION. */
    private static final String GEOPACKAGE_FILE_EXTENSION = "gpkg";

    /** The Constant SPATIALITE_FILE_EXTENSION. */
    private static final String SPATIALITE_FILE_EXTENSION = "db";

    /** The Constant DATABASE_TYPE_KEY. */
    public static final String DATABASE_TYPE_KEY = JDBCDataStoreFactory.DBTYPE.key;

    /** The name map. */
    private static Map<String, String> nameMap = new HashMap<>();

    /** The file handler list. */
    private static List<FileHandlerInterface> fileHandlerList = new ArrayList<>();

    /** The file handler map. */
    private static Map<DatabaseFileHandler, String> fileHandlerMap = new HashMap<>();

    /** Default private constructor */
    private DatabaseConnectionFactory() {
        // Default private constructor
    }

    /**
     * Creates a new DatabaseConnection object for a GeoPackage.
     *
     * @return the database connection
     */
    public static DatabaseConnection createGeoPackage() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        FileNameExtensionFilter filter =
                new FileNameExtensionFilter(
                        Localisation.getString(
                                        DatabaseConnector.class,
                                        "DatabaseConnectorGeoPkg.fileExtension")
                                + " (*."
                                + GEOPACKAGE_FILE_EXTENSION
                                + ")",
                        GEOPACKAGE_FILE_EXTENSION);

        list.add(new DatabaseConnectionField(GeoPkgDataStoreFactory.DATABASE, filter));
        list.add(new DatabaseConnectionField(GeoPkgDataStoreFactory.USER));

        GeoPkgDataStoreFactory factory = new GeoPkgDataStoreFactory();
        return new DatabaseConnection(
                GeoPkgDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                false,
                list,
                new DatabaseConnectionNameInterface() {

                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                Localisation.getString(
                                        DatabaseConnectionFactory.class,
                                        Localisation.COMMON_NOT_SET);
                        String databaseName = properties.get(JDBCDataStoreFactory.DATABASE.key);
                        if (databaseName != null) {
                            File f = new File(databaseName);
                            if (f.isFile()) {
                                connectionName = f.getName();
                            }
                        }

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Adds the file database.
     *
     * @param databaseFileHandler the database file handler
     * @param param the param
     */
    private static void addFileDatabase(DatabaseFileHandler databaseFileHandler, Param param) {
        fileHandlerList.add(databaseFileHandler);
        fileHandlerMap.put(databaseFileHandler, (String) param.sample);
    }

    /**
     * Creates a new DatabaseConnection object for a GeoPackage.
     *
     * @return the database connection
     */
    public static DatabaseConnection createSpatiaLite() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(SpatiaLiteDataStoreFactory.DATABASE));

        FileNameExtensionFilter filter =
                new FileNameExtensionFilter(
                        Localisation.getString(
                                        DatabaseConnector.class,
                                        "DatabaseConnectorGeoPkg.fileExtension")
                                + " (*."
                                + SPATIALITE_FILE_EXTENSION
                                + ")",
                        SPATIALITE_FILE_EXTENSION);
        list.add(new DatabaseConnectionField(SpatiaLiteDataStoreFactory.USER, filter));

        SpatiaLiteDataStoreFactory factory = new SpatiaLiteDataStoreFactory();

        return new DatabaseConnection(
                SpatiaLiteDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                false,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                Localisation.getString(
                                        DatabaseConnectionFactory.class,
                                        Localisation.COMMON_NOT_SET);
                        String databaseName = properties.get(JDBCDataStoreFactory.DATABASE.key);
                        if (databaseName != null) {
                            File f = new File(databaseName);
                            if (f.isFile()) {
                                connectionName = f.getName();
                            }
                        }

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for DB2.
     *
     * @return the database connection
     */
    public static DatabaseConnection createDB2() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(DB2NGDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(DB2NGDataStoreFactory.PORT));
        Param tabSchema = new Param("tabschema", String.class, "Schema", false);
        list.add(new DatabaseConnectionField(tabSchema));
        list.add(new DatabaseConnectionField(DB2NGDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(DB2NGDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(DB2NGDataStoreFactory.PASSWD));

        DB2NGDataStoreFactory factory = new DB2NGDataStoreFactory();

        return new DatabaseConnection(
                DB2NGDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        CONNECTION_STRING_4,
                                        properties.get("tabschema"),
                                        properties.get(JDBCDataStoreFactory.DATABASE.key),
                                        properties.get(JDBCDataStoreFactory.HOST.key),
                                        properties.get(JDBCDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for H2.
     *
     * @return the database connection
     */
    public static DatabaseConnection createH2() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(H2DataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(H2DataStoreFactory.USER));

        H2DataStoreFactory factory = new H2DataStoreFactory();

        return new DatabaseConnection(
                H2DataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                false,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                Localisation.getString(
                                        DatabaseConnectionFactory.class,
                                        Localisation.COMMON_NOT_SET);
                        String databaseName = properties.get(JDBCDataStoreFactory.DATABASE.key);
                        if (databaseName != null) {
                            File f = new File(databaseName);
                            if (f.isFile()) {
                                connectionName = f.getName();
                            }
                        }

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for MySQL.
     *
     * @return the database connection
     */
    public static DatabaseConnection createMySQL() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.PORT));
        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.STORAGE_ENGINE));
        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(MySQLDataStoreFactory.PASSWD));

        MySQLDataStoreFactory factory = new MySQLDataStoreFactory();

        return new DatabaseConnection(
                MySQLDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        "%[email protected]%s:%s",
                                        properties.get(MySQLDataStoreFactory.DATABASE.key),
                                        properties.get(MySQLDataStoreFactory.HOST.key),
                                        properties.get(MySQLDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object.
     *
     * @param connectionName the connection name
     * @param noOfTimesDuplicated the no of times duplicated
     * @param duplicatePrefix the duplicate prefix
     * @return the string
     */
    private static String createConnectionName(
            String connectionName, int noOfTimesDuplicated, String duplicatePrefix) {
        StringBuilder sb = new StringBuilder(connectionName);

        for (int i = 0; i < noOfTimesDuplicated; i++) {
            sb.insert(0, duplicatePrefix);
        }
        return sb.toString();
    }

    /**
     * Creates a new DatabaseConnection object for Oracle.
     *
     * @return the database connection
     */
    public static DatabaseConnection createOracle() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.PORT));
        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.SCHEMA));
        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(OracleNGDataStoreFactory.PASSWD));

        OracleNGDataStoreFactory factory = new OracleNGDataStoreFactory();

        return new DatabaseConnection(
                OracleNGDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        CONNECTION_STRING_4,
                                        properties.get(OracleNGDataStoreFactory.SCHEMA.key),
                                        properties.get(OracleNGDataStoreFactory.DATABASE.key),
                                        properties.get(OracleNGDataStoreFactory.HOST.key),
                                        properties.get(OracleNGDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for SQL Server.
     *
     * @return the database connection
     */
    public static DatabaseConnection createSQLServer() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.PORT));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.INSTANCE));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.SCHEMA));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.PASSWD));
        list.add(
                new DatabaseConnectionField(JTDSSqlServerDataStoreFactory.GEOMETRY_METADATA_TABLE));

        JDTSSQLServerJNDIDataStoreFactory factory = new JDTSSQLServerJNDIDataStoreFactory();

        return new DatabaseConnection(
                JTDSSqlServerDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        CONNECTION_STRING_4,
                                        properties.get(JTDSSqlServerDataStoreFactory.INSTANCE.key),
                                        properties.get(JTDSSqlServerDataStoreFactory.DATABASE.key),
                                        properties.get(JTDSSqlServerDataStoreFactory.HOST.key),
                                        properties.get(JTDSSqlServerDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for Teradata.
     *
     * @return the database connection
     */
    public static DatabaseConnection createTeradata() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.PORT));
        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.SCHEMA));
        list.add(new DatabaseConnectionField(TeradataDataStoreFactory.PASSWD));

        TeradataDataStoreFactory factory = new TeradataDataStoreFactory();

        return new DatabaseConnection(
                TeradataDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        CONNECTION_STRING_4,
                                        properties.get(TeradataDataStoreFactory.SCHEMA.key),
                                        properties.get(TeradataDataStoreFactory.DATABASE.key),
                                        properties.get(TeradataDataStoreFactory.HOST.key),
                                        properties.get(TeradataDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Creates a new DatabaseConnection object for Postgres.
     *
     * @return the database connection
     */
    public static DatabaseConnection createPostgres() {
        List<DatabaseConnectionField> list = new ArrayList<>();

        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.HOST));
        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.PORT));
        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.DATABASE));
        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.SCHEMA));
        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.USER));
        list.add(new DatabaseConnectionField(PostgisNGDataStoreFactory.PASSWD));

        PostgisNGDataStoreFactory factory = new PostgisNGDataStoreFactory();

        return new DatabaseConnection(
                PostgisNGDataStoreFactory.DBTYPE,
                factory.getDisplayName(),
                true,
                list,
                new DatabaseConnectionNameInterface() {
                    /** The Constant serialVersionUID. */
                    private static final long serialVersionUID = 1L;

                    @Override
                    public String getConnectionName(
                            String duplicatePrefix,
                            int noOfTimesDuplicated,
                            Map<String, String> properties) {
                        String connectionName =
                                String.format(
                                        CONNECTION_STRING_4,
                                        properties.get(PostgisNGDataStoreFactory.SCHEMA.key),
                                        properties.get(PostgisNGDataStoreFactory.DATABASE.key),
                                        properties.get(PostgisNGDataStoreFactory.HOST.key),
                                        properties.get(PostgisNGDataStoreFactory.PORT.key));

                        return createConnectionName(
                                connectionName, noOfTimesDuplicated, duplicatePrefix);
                    }
                });
    }

    /**
     * Decode string.
     *
     * @param localConnectionDataMap the local connection data map
     * @return the database connection
     */
    public static DatabaseConnection decodeString(Map<String, String> localConnectionDataMap) {
        if (localConnectionDataMap == null) {
            return null;
        }

        String type = localConnectionDataMap.get(DATABASE_TYPE_KEY);

        return createDefault(type);
    }

    /**
     * Creates a new DatabaseConnection object.
     *
     * @param type the type
     * @return the database connection
     */
    private static DatabaseConnection createDefault(String type) {
        if (type != null) {
            if (type.equals(GeoPkgDataStoreFactory.DBTYPE.sample)) {
                return createGeoPackage();
            } else if (type.equals(PostgisNGDataStoreFactory.DBTYPE.sample)) {
                return createPostgres();
            } else if (type.equals(SpatiaLiteDataStoreFactory.DBTYPE.sample)) {
                return createSpatiaLite();
            } else if (type.equals(TeradataDataStoreFactory.DBTYPE.sample)) {
                return createTeradata();
            } else if (type.equals(JTDSSqlServerDataStoreFactory.DBTYPE.sample)) {
                return createSQLServer();
            } else if (type.equals(SQLServerDataStoreFactory.DBTYPE.sample)) {
                return createSQLServer();
            } else if (type.equals(OracleNGDataStoreFactory.DBTYPE.sample)) {
                return createOracle();
            } else if (type.equals(MySQLDataStoreFactory.DBTYPE.sample)) {
                return createMySQL();
            } else if (type.equals(H2DataStoreFactory.DBTYPE.sample)) {
                return createH2();
            } else if (type.equals(DB2NGDataStoreFactory.DBTYPE.sample)) {
                return createDB2();
            }
        }
        return null;
    }

    /**
     * Gets the list of database connection names.
     *
     * @return the names
     */
    public static List<String> getNames() {
        List<String> list = new ArrayList<>();

        if (nameMap.isEmpty()) {
            populateNameMap();
        }

        for (String key : nameMap.keySet()) {
            list.add(key);
        }
        return list;
    }

    /** Populate name map. */
    private static void populateNameMap() {
        Iterator<DataStoreFactorySpi> datastore = DataStoreFinder.getAvailableDataStores();

        while (datastore.hasNext()) {
            DataStoreFactorySpi dSPI = datastore.next();

            Param dbType = null;
            for (Param param : dSPI.getParametersInfo()) {
                if (param.key.equals(JDBCDataStoreFactory.DBTYPE.key)) {
                    dbType = param;
                    break;
                }
            }
            if (dbType != null) {
                nameMap.put(dSPI.getDisplayName(), (String) dbType.sample);
            }
        }
    }

    /**
     * Gets the new connection.
     *
     * @param selectedDatabaseType the selected database type
     * @return the new connection
     */
    public static DatabaseConnection getNewConnection(String selectedDatabaseType) {
        if (nameMap.isEmpty()) {
            populateNameMap();
        }
        String type = nameMap.get(selectedDatabaseType);
        return createDefault(type);
    }

    /**
     * Gets the new connection.
     *
     * @param databaseConnection the database connection
     * @return the new connection
     */
    public static DatabaseConnection getNewConnection(DatabaseConnection databaseConnection) {
        return createDefault(databaseConnection.getDatabaseType());
    }

    /**
     * Gets the database connection.
     *
     * @param filename the filename
     * @return the new connection
     */
    public static DatabaseConnection getConnection(String filename) {
        List<FileHandlerInterface> list = getFileHandlers();

        if (filename != null) {
            for (FileHandlerInterface handler : list) {
                for (String fileExtension : handler.getFileExtensionList()) {
                    if (filename.endsWith(fileExtension)) {
                        String dbConnectionType = fileHandlerMap.get(handler);

                        DatabaseConnection dbConnection = createDefault(dbConnectionType);

                        if (dbConnection != null) {
                            Map<String, String> connectionDataMap = new HashMap<>();

                            connectionDataMap.put(JDBCDataStoreFactory.DATABASE.key, filename);
                            dbConnection.setConnectionDataMap(connectionDataMap);
                        }
                        return dbConnection;
                    }
                }
            }
        }
        return null;
    }

    /**
     * Gets the file handlers.
     *
     * @return the file handlers
     */
    public static List<FileHandlerInterface> getFileHandlers() {
        if (fileHandlerList.isEmpty()) {
            addFileDatabase(
                    new DatabaseFileHandler(
                            "ui/filesystemicons/spatialite.png",
                            Arrays.asList(SPATIALITE_FILE_EXTENSION)),
                    SpatiaLiteDataStoreFactory.DBTYPE);

            addFileDatabase(
                    new DatabaseFileHandler(
                            "ui/filesystemicons/geopackage.png",
                            Arrays.asList(GEOPACKAGE_FILE_EXTENSION)),
                    GeoPkgDataStoreFactory.DBTYPE);
        }
        return fileHandlerList;
    }
}