How do I create an ODBC DSN entry using C#?

前端 未结 7 1203
旧巷少年郎
旧巷少年郎 2020-11-29 23:43

I\'m working on a legacy application that has a C++ extended stored procedure. This xsproc uses ODBC to connect to the database, which means it requires a DSN to be config

相关标签:
7条回答
  • 2020-11-29 23:50

    +1 for Barnwell's code!

    However, I think his DSNExists() is querying the wrong key. I think it should be this:

    public static bool DSNExists(string dsnName) 
    { 
        var sourcesKey = Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + "ODBC Data Sources"); 
        if (sourcesKey == null) throw new Exception("ODBC Registry key for sources does not exist"); 
    
        return sourcesKey.GetValue(dsnName) != null; 
    } 
    
    0 讨论(0)
  • 2020-11-29 23:53

    There is a CodeProject page on reading ODBC information.

    Reading that should give you the information you need to reverse engineer writing the registry entries you need.

    From that code;

      private const string ODBC_LOC_IN_REGISTRY = "SOFTWARE\\ODBC\\";
      private const string ODBC_INI_LOC_IN_REGISTRY =
              ODBC_LOC_IN_REGISTRY + "ODBC.INI\\";
    
      private const string DSN_LOC_IN_REGISTRY =
              ODBC_INI_LOC_IN_REGISTRY + "ODBC Data Sources\\";
    
      private const string ODBCINST_INI_LOC_IN_REGISTRY =
              ODBC_LOC_IN_REGISTRY + "ODBCINST.INI\\";
    
      private const string ODBC_DRIVERS_LOC_IN_REGISTRY =
              ODBCINST_INI_LOC_IN_REGISTRY + "ODBC Drivers\\";
    
    0 讨论(0)
  • 2020-11-29 23:59

    thanks it was a great help if you are making a dsn to excel maybe need add something like this

     var dsnKeyEng = Microsoft.Win32.Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + dsnName + "\\Engines");
     var dsnKeyExl = Microsoft.Win32.Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + dsnName + "\\Engines\\Excel");
    
     dsnKeyExl.SetValue("FirstRowHasNames", 01);
     dsnKeyExl.SetValue("MaxScanRows", 8);
     dsnKeyExl.SetValue("Threads",3);
     dsnKeyExl.SetValue("UserCommitSync", "Yes")
    
    0 讨论(0)
  • 2020-11-30 00:00

    There is an API for doing stuff like this. Using the API will also make sure that your application will stay compatible with newer versions of Windows. The API can be found here:

    http://msdn.microsoft.com/en-us/library/ms716476(VS.85).aspx

    PInvoking this function in c# can be found on PInvoke.net.

    0 讨论(0)
  • 2020-11-30 00:06

    Thanks for providing this code, I have used it myself. I had to change two things tough:

    To get the driverName I had to use OpenSubKey instead of CreateSubKey to get the values:

    // Lookup driver path from driver name
    var driverKey = Microsoft.Win32.Registry.LocalMachine.OpenSubKey(
            ODBCINST_INI_REG_PATH + driverName);
    

    Since I am running Vista, I had to use an application manifest and set the requestedPrivileges to:

    <requestedExecutionLevel level="requireAdministrator" uiAccess="false"/>
    

    The following article helped me to find the OpenSubKey issue: http://www.daveoncsharp.com/2009/08/read-write-delete-from-windows-registry-with-csharp/

    0 讨论(0)
  • 2020-11-30 00:08

    I actually solved this myself in the end by manipulating the registry. I've created a class to contain the functionality, the contents of which I've included here:

    ///<summary>
    /// Class to assist with creation and removal of ODBC DSN entries
    ///</summary>
    public static class ODBCManager
    {
        private const string ODBC_INI_REG_PATH = "SOFTWARE\\ODBC\\ODBC.INI\\";
        private const string ODBCINST_INI_REG_PATH = "SOFTWARE\\ODBC\\ODBCINST.INI\\";
    
        /// <summary>
        /// Creates a new DSN entry with the specified values. If the DSN exists, the values are updated.
        /// </summary>
        /// <param name="dsnName">Name of the DSN for use by client applications</param>
        /// <param name="description">Description of the DSN that appears in the ODBC control panel applet</param>
        /// <param name="server">Network name or IP address of database server</param>
        /// <param name="driverName">Name of the driver to use</param>
        /// <param name="trustedConnection">True to use NT authentication, false to require applications to supply username/password in the connection string</param>
        /// <param name="database">Name of the datbase to connect to</param>
        public static void CreateDSN(string dsnName, string description, string server, string driverName, bool trustedConnection, string database)
        {
            // Lookup driver path from driver name
            var driverKey = Registry.LocalMachine.CreateSubKey(ODBCINST_INI_REG_PATH + driverName);
            if (driverKey == null) throw new Exception(string.Format("ODBC Registry key for driver '{0}' does not exist", driverName));
            string driverPath = driverKey.GetValue("Driver").ToString();
    
            // Add value to odbc data sources
            var datasourcesKey = Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + "ODBC Data Sources");
            if (datasourcesKey == null) throw new Exception("ODBC Registry key for datasources does not exist");
            datasourcesKey.SetValue(dsnName, driverName);
    
            // Create new key in odbc.ini with dsn name and add values
            var dsnKey = Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + dsnName);
            if (dsnKey == null) throw new Exception("ODBC Registry key for DSN was not created");
            dsnKey.SetValue("Database", database);
            dsnKey.SetValue("Description", description);
            dsnKey.SetValue("Driver", driverPath);
            dsnKey.SetValue("LastUser", Environment.UserName);
            dsnKey.SetValue("Server", server);
            dsnKey.SetValue("Database", database);
            dsnKey.SetValue("Trusted_Connection", trustedConnection ? "Yes" : "No");
        }
    
        /// <summary>
        /// Removes a DSN entry
        /// </summary>
        /// <param name="dsnName">Name of the DSN to remove.</param>
        public static void RemoveDSN(string dsnName)
        {
            // Remove DSN key
            Registry.LocalMachine.DeleteSubKeyTree(ODBC_INI_REG_PATH + dsnName);
    
            // Remove DSN name from values list in ODBC Data Sources key
            var datasourcesKey = Registry.LocalMachine.CreateSubKey(ODBC_INI_REG_PATH + "ODBC Data Sources");
            if (datasourcesKey == null) throw new Exception("ODBC Registry key for datasources does not exist");
            datasourcesKey.DeleteValue(dsnName);
        }
    
        ///<summary>
        /// Checks the registry to see if a DSN exists with the specified name
        ///</summary>
        ///<param name="dsnName"></param>
        ///<returns></returns>
        public static bool DSNExists(string dsnName)
        {
            var driversKey = Registry.LocalMachine.CreateSubKey(ODBCINST_INI_REG_PATH + "ODBC Drivers");
            if (driversKey == null) throw new Exception("ODBC Registry key for drivers does not exist");
    
            return driversKey.GetValue(dsnName) != null;
        }
    
        ///<summary>
        /// Returns an array of driver names installed on the system
        ///</summary>
        ///<returns></returns>
        public static string[] GetInstalledDrivers()
        {
            var driversKey = Registry.LocalMachine.CreateSubKey(ODBCINST_INI_REG_PATH + "ODBC Drivers");
            if (driversKey == null) throw new Exception("ODBC Registry key for drivers does not exist");
    
            var driverNames = driversKey.GetValueNames();
    
            var ret = new List<string>();
    
            foreach (var driverName in driverNames)
            {
                if (driverName != "(Default)")
                {
                    ret.Add(driverName);
                }
            }
    
            return ret.ToArray();
        }
    }
    
    0 讨论(0)
提交回复
热议问题