746 lines
31 KiB
C#
746 lines
31 KiB
C#
|
//------------------------------------------------------------------------------
|
||
|
// <copyright file="Configuration.cs" company="Microsoft">
|
||
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
// </copyright>
|
||
|
//------------------------------------------------------------------------------
|
||
|
|
||
|
namespace System.Web.Management {
|
||
|
using System;
|
||
|
using System.Web.Util;
|
||
|
using System.IO;
|
||
|
using System.Data;
|
||
|
using System.Data.SqlClient;
|
||
|
using System.Threading;
|
||
|
using System.Text.RegularExpressions;
|
||
|
using System.Text;
|
||
|
using System.Security;
|
||
|
using System.Security.Permissions;
|
||
|
using System.Globalization;
|
||
|
using System.Runtime.Serialization;
|
||
|
using System.Collections;
|
||
|
|
||
|
|
||
|
[Flags]
|
||
|
public enum SqlFeatures : int {
|
||
|
None = 0x00000000,
|
||
|
Membership = 0x00000001,
|
||
|
Profile = 0x00000002,
|
||
|
RoleManager = 0x00000004,
|
||
|
Personalization = 0x00000008,
|
||
|
SqlWebEventProvider= 0x00000010,
|
||
|
|
||
|
// We need to add 0x40000000 to All in order to differentiate between All, and
|
||
|
// an combination of all other flags. Because if the user pass in All, we have
|
||
|
// to remove the common tables too.
|
||
|
All = Membership|Profile|RoleManager|
|
||
|
Personalization|SqlWebEventProvider|(int)0x40000000,
|
||
|
};
|
||
|
|
||
|
public enum SessionStateType {
|
||
|
Temporary,
|
||
|
Persisted,
|
||
|
Custom,
|
||
|
}
|
||
|
|
||
|
[Serializable()]
|
||
|
public sealed class SqlExecutionException : SystemException {
|
||
|
string _server;
|
||
|
string _database;
|
||
|
string _sqlFile;
|
||
|
string _commands;
|
||
|
SqlException _sqlException;
|
||
|
|
||
|
public SqlExecutionException(string message, string server, string database,
|
||
|
string sqlFile, string commands, SqlException sqlException)
|
||
|
: base(message) {
|
||
|
_server = server;
|
||
|
_database = database;
|
||
|
_sqlFile = sqlFile;
|
||
|
_commands = commands;
|
||
|
_sqlException = sqlException;
|
||
|
}
|
||
|
|
||
|
public SqlExecutionException(String message)
|
||
|
: base(message) {
|
||
|
}
|
||
|
|
||
|
public SqlExecutionException(string message, Exception innerException)
|
||
|
: base (message, innerException) {
|
||
|
}
|
||
|
|
||
|
public SqlExecutionException() {
|
||
|
}
|
||
|
|
||
|
private SqlExecutionException(SerializationInfo info, StreamingContext context)
|
||
|
:base(info, context) {
|
||
|
_server = info.GetString("_server");
|
||
|
_database = info.GetString("_database");
|
||
|
_sqlFile = info.GetString("_sqlFile");
|
||
|
_commands = info.GetString("_commands");
|
||
|
_sqlException = (SqlException)info.GetValue("_sqlException", typeof(SqlException));
|
||
|
}
|
||
|
|
||
|
[SecurityPermissionAttribute(SecurityAction.Demand, SerializationFormatter=true)]
|
||
|
public override void GetObjectData(SerializationInfo info, StreamingContext context)
|
||
|
{
|
||
|
base.GetObjectData(info, context);
|
||
|
info.AddValue("_server", _server);
|
||
|
info.AddValue("_database", _database);
|
||
|
info.AddValue("_sqlFile", _sqlFile);
|
||
|
info.AddValue("_commands", _commands);
|
||
|
info.AddValue("_sqlException", _sqlException);
|
||
|
}
|
||
|
|
||
|
// Name of the server we're connecting to
|
||
|
public string Server {
|
||
|
get { return _server; }
|
||
|
}
|
||
|
|
||
|
// Name of database we use
|
||
|
public string Database {
|
||
|
get { return _database; }
|
||
|
}
|
||
|
|
||
|
// Name of the SQL file we load
|
||
|
public string SqlFile {
|
||
|
get { return _sqlFile; }
|
||
|
}
|
||
|
|
||
|
// The set of batched commands that failed
|
||
|
public string Commands {
|
||
|
get { return _commands; }
|
||
|
}
|
||
|
|
||
|
// The SQL exeception from the failure
|
||
|
public SqlException Exception {
|
||
|
get { return _sqlException; }
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[AspNetHostingPermission(SecurityAction.LinkDemand, Level=AspNetHostingPermissionLevel.High)]
|
||
|
public static class SqlServices {
|
||
|
public static void Install(string server, string user, string password, string database, SqlFeatures features) {
|
||
|
SetupApplicationServices(server, user, password, false, null, database, null, features, true);
|
||
|
}
|
||
|
|
||
|
// For trusted connection
|
||
|
public static void Install(string server, string database, SqlFeatures features) {
|
||
|
SetupApplicationServices(server, null, null, true, null, database, null, features, true);
|
||
|
}
|
||
|
|
||
|
internal static void Install( string database, string dbFileName, string connectionString ) {
|
||
|
SetupApplicationServices( null, null, null, false, connectionString, database, dbFileName, SqlFeatures.All, true );
|
||
|
}
|
||
|
|
||
|
// For connection using connection string
|
||
|
public static void Install(string database, SqlFeatures features, string connectionString) {
|
||
|
SetupApplicationServices(null, null, null, true, connectionString, database, null, features, true);
|
||
|
}
|
||
|
|
||
|
public static void Uninstall(string server, string user, string password, string database, SqlFeatures features) {
|
||
|
SetupApplicationServices(server, user, password, false, null, database, null, features, false);
|
||
|
}
|
||
|
|
||
|
// For trusted connection
|
||
|
public static void Uninstall(string server, string database, SqlFeatures features) {
|
||
|
SetupApplicationServices(server, null, null, true, null, database, null, features, false);
|
||
|
}
|
||
|
|
||
|
// For connection using connection string
|
||
|
public static void Uninstall(string database, SqlFeatures features, string connectionString) {
|
||
|
SetupApplicationServices(null, null, null, true, connectionString, database, null, features, false);
|
||
|
}
|
||
|
|
||
|
public static void InstallSessionState(string server, string user, string password, string customDatabase, SessionStateType type) {
|
||
|
SetupSessionState(server, user, password, false, null, customDatabase, type, true);
|
||
|
}
|
||
|
|
||
|
// For trusted connection
|
||
|
public static void InstallSessionState(string server, string customDatabase, SessionStateType type) {
|
||
|
SetupSessionState(server, null, null, true, null, customDatabase, type, true);
|
||
|
}
|
||
|
|
||
|
// For connection using connection string
|
||
|
public static void InstallSessionState(string customDatabase, SessionStateType type, string connectionString) {
|
||
|
SetupSessionState(null, null, null, true, connectionString, customDatabase, type, true);
|
||
|
}
|
||
|
|
||
|
|
||
|
public static void UninstallSessionState(string server, string user, string password, string customDatabase, SessionStateType type) {
|
||
|
SetupSessionState(server, user, password, false, null, customDatabase, type, false);
|
||
|
}
|
||
|
|
||
|
// For trusted connection
|
||
|
public static void UninstallSessionState(string server, string customDatabase, SessionStateType type) {
|
||
|
SetupSessionState(server, null, null, true, null, customDatabase, type, false);
|
||
|
}
|
||
|
|
||
|
// For connection using connection string
|
||
|
public static void UninstallSessionState(string customDatabase, SessionStateType type, string connectionString) {
|
||
|
SetupSessionState(null, null, null, true, connectionString, customDatabase, type, false);
|
||
|
}
|
||
|
|
||
|
// Used by suite
|
||
|
internal static ArrayList ApplicationServiceTables {
|
||
|
get {
|
||
|
ArrayList tables = new ArrayList();
|
||
|
for (int i=0; i < s_featureInfos.Length; i++) {
|
||
|
tables.InsertRange(tables.Count, s_featureInfos[i]._tablesRemovedInUninstall);
|
||
|
}
|
||
|
return tables;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static string GenerateSessionStateScripts(bool install, SessionStateType type, string customDatabase) {
|
||
|
SessionStateParamCheck(type, ref customDatabase);
|
||
|
string fullpath = Path.Combine(HttpRuntime.AspInstallDirectory, install ? SESSION_STATE_INSTALL_FILE : SESSION_STATE_UNINSTALL_FILE);
|
||
|
string content = File.ReadAllText(fullpath);
|
||
|
return FixContent(content, customDatabase, null, true, type);
|
||
|
}
|
||
|
|
||
|
|
||
|
// Information about a feature.
|
||
|
internal struct FeatureInfo {
|
||
|
internal SqlFeatures _feature;
|
||
|
internal string[] _installFiles; // SQL file to run to install the feature
|
||
|
internal string[] _uninstallFiles; // SQL file to run to uninstall the feature
|
||
|
internal string[] _tablesRemovedInUninstall; // SQL tables to be removed during uninstall
|
||
|
internal int _dataCheckBitMask;
|
||
|
|
||
|
internal FeatureInfo(SqlFeatures feature, string[] installFiles,
|
||
|
string[] uninstallFiles, string[] tablesRemovedInUninstall, int dataCheckBitMask) {
|
||
|
_feature = feature;
|
||
|
_installFiles = installFiles;
|
||
|
_uninstallFiles = uninstallFiles;
|
||
|
_tablesRemovedInUninstall = tablesRemovedInUninstall;
|
||
|
_dataCheckBitMask = dataCheckBitMask;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static string INSTALL_COMMON_SQL = "InstallCommon.sql";
|
||
|
|
||
|
static FeatureInfo[] s_featureInfos = {
|
||
|
new FeatureInfo(SqlFeatures.Membership,
|
||
|
new string[] {INSTALL_COMMON_SQL, "InstallMembership.sql"},
|
||
|
new string[] {"UninstallMembership.sql"},
|
||
|
new string[] {"aspnet_Membership"},
|
||
|
1),
|
||
|
|
||
|
new FeatureInfo(SqlFeatures.Profile,
|
||
|
new string[] {INSTALL_COMMON_SQL, "InstallProfile.sql"},
|
||
|
new string[] {"UninstallProfile.sql"},
|
||
|
new string[] {"aspnet_Profile"},
|
||
|
4),
|
||
|
|
||
|
new FeatureInfo(SqlFeatures.RoleManager,
|
||
|
new string[] {INSTALL_COMMON_SQL, "InstallRoles.sql"},
|
||
|
new string[] {"UninstallRoles.sql"},
|
||
|
new string[] {"aspnet_Roles", "aspnet_UsersInRoles"},
|
||
|
2),
|
||
|
|
||
|
new FeatureInfo(SqlFeatures.Personalization,
|
||
|
new string[] {INSTALL_COMMON_SQL, "InstallPersonalization.sql"},
|
||
|
new string[] {"UninstallPersonalization.sql"},
|
||
|
new string[] {"aspnet_PersonalizationPerUser", "aspnet_Paths", "aspnet_PersonalizationAllUsers"},
|
||
|
8),
|
||
|
|
||
|
new FeatureInfo(SqlFeatures.SqlWebEventProvider,
|
||
|
new string[] {INSTALL_COMMON_SQL, "InstallWebEventSqlProvider.sql"},
|
||
|
new string[] {"UninstallWebEventSqlProvider.sql"},
|
||
|
new string[] {"aspnet_WebEvent_Events"},
|
||
|
16),
|
||
|
|
||
|
// The following are the files to run in addition to those listed in other feature
|
||
|
new FeatureInfo(SqlFeatures.All,
|
||
|
new string[] {},
|
||
|
new string[] {"UninstallCommon.sql"},
|
||
|
new string[] {"aspnet_Applications", "aspnet_Users", "aspnet_SchemaVersions"},
|
||
|
0x7FFFFFFF),
|
||
|
};
|
||
|
|
||
|
// Default database to use if the database name isn't supplied by the user
|
||
|
static string DEFAULT_DB = "aspnetdb";
|
||
|
static string ASPSTATE_DB = "ASPState";
|
||
|
static string SSTYPE_PERSISTED = "sstype_persisted";
|
||
|
static string SSTYPE_CUSTOM = "sstype_custom";
|
||
|
|
||
|
static string SESSION_STATE_INSTALL_FILE = "InstallSqlState.sql";
|
||
|
static string SESSION_STATE_UNINSTALL_FILE = "UninstallSqlState.sql";
|
||
|
|
||
|
// Return a list of files based on install (bool) and features
|
||
|
static ArrayList GetFiles(bool install, SqlFeatures features) {
|
||
|
ArrayList results = new ArrayList();
|
||
|
bool installCommonProcessed = false;
|
||
|
|
||
|
// Load and modify the sql file for each feature
|
||
|
for (int i=0; i < s_featureInfos.Length; i++) {
|
||
|
string[] sqlFiles = null;
|
||
|
|
||
|
if (((int)s_featureInfos[i]._feature & (int)features) == (int)s_featureInfos[i]._feature) {
|
||
|
// We found one feature
|
||
|
if (install) {
|
||
|
sqlFiles = s_featureInfos[i]._installFiles;
|
||
|
}
|
||
|
else {
|
||
|
sqlFiles = s_featureInfos[i]._uninstallFiles;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (sqlFiles != null) {
|
||
|
for(int j = 0; j < sqlFiles.Length; j++) {
|
||
|
string sqlFile = sqlFiles[j];
|
||
|
if (sqlFile != null) {
|
||
|
// We only need to process InstallCommon.sql once
|
||
|
if (sqlFile == INSTALL_COMMON_SQL && installCommonProcessed) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
results.Add(sqlFile);
|
||
|
|
||
|
if (!installCommonProcessed && sqlFile == INSTALL_COMMON_SQL) {
|
||
|
installCommonProcessed = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return results;
|
||
|
}
|
||
|
|
||
|
// Replace the name of the database with the one specified by the caller
|
||
|
static string FixContent(
|
||
|
string content,
|
||
|
string database,
|
||
|
string dbFileName,
|
||
|
bool sessionState,
|
||
|
SessionStateType sessionStatetype
|
||
|
)
|
||
|
{
|
||
|
if (database != null) {
|
||
|
database = RemoveSquareBrackets(database);
|
||
|
}
|
||
|
|
||
|
if (sessionState) {
|
||
|
if (sessionStatetype == SessionStateType.Temporary) {
|
||
|
// No change
|
||
|
}
|
||
|
else if (sessionStatetype == SessionStateType.Persisted) {
|
||
|
content = content.Replace("'sstype_temp'", "'" + SSTYPE_PERSISTED + "'");
|
||
|
content = content.Replace("[tempdb]", "[" + ASPSTATE_DB + "]");
|
||
|
}
|
||
|
else if (sessionStatetype == SessionStateType.Custom) {
|
||
|
content = content.Replace("'sstype_temp'", "'" + SSTYPE_CUSTOM + "'");
|
||
|
content = content.Replace("[tempdb]", "[" + database + "]");
|
||
|
content = content.Replace("'ASPState'", "'" + database + "'");
|
||
|
content = content.Replace("[ASPState]", "[" + database + "]");
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
content = content.Replace("'aspnetdb'", "'" + database.Replace("'", "''") + "'");
|
||
|
content = content.Replace("[aspnetdb]", "[" + database + "]");
|
||
|
}
|
||
|
|
||
|
if( dbFileName != null )
|
||
|
{
|
||
|
if (dbFileName.Contains("[") || dbFileName.Contains("]") || dbFileName.Contains("'"))
|
||
|
throw new ArgumentException(SR.GetString(SR.DbFileName_can_not_contain_invalid_chars));
|
||
|
database = database.TrimStart( '[' );
|
||
|
database = database.TrimEnd( ']' );
|
||
|
|
||
|
string logicalFileName = database + "_DAT";
|
||
|
if (!char.IsLetter(logicalFileName[0]))
|
||
|
logicalFileName = "A" + logicalFileName;
|
||
|
|
||
|
//
|
||
|
// Build the database options string for SQL Express database
|
||
|
//
|
||
|
string dbOptions = "ON ( NAME = " + logicalFileName + ", FILENAME = ''" +
|
||
|
dbFileName + "'', " + "SIZE = 10MB, FILEGROWTH = 5MB )";
|
||
|
|
||
|
content = content.Replace("SET @dboptions = N'/**/'",
|
||
|
"SET @dboptions = N'" + dbOptions + "'");
|
||
|
}
|
||
|
|
||
|
return content;
|
||
|
}
|
||
|
|
||
|
static void ExecuteSessionFile(
|
||
|
string file,
|
||
|
string server,
|
||
|
string database,
|
||
|
string dbFileName,
|
||
|
SqlConnection connection,
|
||
|
bool isInstall,
|
||
|
SessionStateType sessionStatetype
|
||
|
) {
|
||
|
ExecuteFile(file, server, database, dbFileName, connection, true, isInstall, sessionStatetype);
|
||
|
}
|
||
|
|
||
|
// Load the SQL file, change the database name within, and execute it.
|
||
|
static void ExecuteFile(
|
||
|
string file,
|
||
|
string server,
|
||
|
string database,
|
||
|
string dbFileName,
|
||
|
SqlConnection connection,
|
||
|
bool sessionState,
|
||
|
bool isInstall,
|
||
|
SessionStateType sessionStatetype
|
||
|
)
|
||
|
{
|
||
|
string fullpath = Path.Combine(HttpRuntime.AspInstallDirectory, file);
|
||
|
string content = File.ReadAllText(fullpath);
|
||
|
StringReader sr;
|
||
|
string cmdText = null;
|
||
|
string cur;
|
||
|
SqlCommand sqlCmd;
|
||
|
|
||
|
Debug.Trace("SqlServices", "Execute File: about to run " + fullpath);
|
||
|
// We need to replace the name of the database with the one specified by the caller
|
||
|
if( file.Equals( INSTALL_COMMON_SQL ) )
|
||
|
{
|
||
|
content = FixContent(content, database, dbFileName, sessionState, sessionStatetype);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
content = FixContent(content, database, null, sessionState, sessionStatetype);
|
||
|
}
|
||
|
|
||
|
sr = new StringReader(content);
|
||
|
|
||
|
sqlCmd = new SqlCommand(null, connection);
|
||
|
do {
|
||
|
bool run = false;
|
||
|
|
||
|
// Read a line from a file. If it's not a "GO", batch it up.
|
||
|
// It it's a "GO" (or the last line), send over all batched
|
||
|
// commands over to SQL.
|
||
|
|
||
|
cur = sr.ReadLine();
|
||
|
|
||
|
if (cur == null) {
|
||
|
run = true;
|
||
|
}
|
||
|
else {
|
||
|
if (StringUtil.EqualsIgnoreCase(cur.Trim(), "GO"))
|
||
|
{
|
||
|
run = true;
|
||
|
}
|
||
|
else {
|
||
|
if (cmdText != null) {
|
||
|
cmdText += "\n";
|
||
|
}
|
||
|
|
||
|
cmdText += cur;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (run & cmdText != null) {
|
||
|
sqlCmd.CommandText = cmdText;
|
||
|
try {
|
||
|
sqlCmd.ExecuteNonQuery();
|
||
|
}
|
||
|
catch (Exception e) {
|
||
|
SqlException sqlExpt = e as SqlException;
|
||
|
|
||
|
if (sqlExpt != null) {
|
||
|
Debug.Trace("SqlServices", "Error executing command. SqlException:" +
|
||
|
"\nMessage=" + sqlExpt.Message +
|
||
|
"\nNumber=" + sqlExpt.Number);
|
||
|
int expectedError = -1;
|
||
|
|
||
|
// There are some errors we might expect.
|
||
|
// See VSWhidbey 376433
|
||
|
|
||
|
if (cmdText.IndexOf("sp_add_category", StringComparison.Ordinal) > -1) {
|
||
|
expectedError = 14261; /* already exists */
|
||
|
}
|
||
|
else if (cmdText.IndexOf("sp_delete_job", StringComparison.Ordinal) > -1) {
|
||
|
expectedError = 14262; /* doesn't exists */
|
||
|
|
||
|
if (sessionState && !isInstall) {
|
||
|
throw new SqlExecutionException(SR.GetString(SR.SQL_Services_Error_Deleting_Session_Job),
|
||
|
server, database, file, cmdText, sqlExpt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (sqlExpt.Number == expectedError) {
|
||
|
Debug.Trace("SqlServices", "Got expected error: " + expectedError + "; not throwing");
|
||
|
}
|
||
|
else {
|
||
|
throw new SqlExecutionException(
|
||
|
SR.GetString(SR.SQL_Services_Error_Executing_Command,
|
||
|
file, sqlExpt.Number.ToString(CultureInfo.CurrentCulture), sqlExpt.Message),
|
||
|
server, database, file, cmdText, sqlExpt);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
#pragma warning disable 1058
|
||
|
catch {
|
||
|
throw;
|
||
|
}
|
||
|
#pragma warning restore 1058
|
||
|
cmdText = null;
|
||
|
}
|
||
|
} while (cur != null);
|
||
|
}
|
||
|
|
||
|
static void ApplicationServicesParamCheck(SqlFeatures features, ref string database) {
|
||
|
if (features == SqlFeatures.None) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ((features & SqlFeatures.All) != features) {
|
||
|
throw new ArgumentException(SR.GetString(SR.SQL_Services_Invalid_Feature));
|
||
|
}
|
||
|
|
||
|
// VSWhidbey 355946:
|
||
|
// SQL will ignore trailing space of database names. It's easier to just
|
||
|
// trim those spaces here than to fix all the SQL statements to handle
|
||
|
// errors due to trailing space.
|
||
|
CheckDatabaseName(ref database);
|
||
|
}
|
||
|
private static void CheckDatabaseName(ref string database)
|
||
|
{
|
||
|
if (database != null) {
|
||
|
database = database.TrimEnd();
|
||
|
|
||
|
if (database.Length == 0)
|
||
|
throw new ArgumentException(SR.GetString(SR.SQL_Services_Database_Empty_Or_Space_Only_Arg));
|
||
|
|
||
|
database = RemoveSquareBrackets(database);
|
||
|
|
||
|
if (database.Contains("'") || database.Contains("[") || database.Contains("]"))
|
||
|
throw new ArgumentException(SR.GetString(SR.SQL_Services_Database_contains_invalid_chars));
|
||
|
}
|
||
|
|
||
|
if (database == null) {
|
||
|
database = DEFAULT_DB;
|
||
|
}
|
||
|
else {
|
||
|
// Wrap it with [] if not already
|
||
|
//if (!(StringUtil.StringStartsWith(database, '[') && StringUtil.StringEndsWith(database, ']'))) {
|
||
|
database = "[" + database + "]";
|
||
|
//}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static string GenerateApplicationServicesScripts(bool install, SqlFeatures features, string database) {
|
||
|
string content;
|
||
|
StringBuilder sb = new StringBuilder();
|
||
|
ArrayList files;
|
||
|
|
||
|
ApplicationServicesParamCheck(features, ref database);
|
||
|
|
||
|
files = GetFiles(install, features);
|
||
|
|
||
|
foreach (string sqlFile in files) {
|
||
|
string fullpath = Path.Combine(HttpRuntime.AspInstallDirectory, sqlFile);
|
||
|
content = File.ReadAllText(fullpath);
|
||
|
sb.Append(FixContent(content, database, null, false, SessionStateType.Temporary));
|
||
|
}
|
||
|
|
||
|
return sb.ToString();
|
||
|
}
|
||
|
|
||
|
static string RemoveSquareBrackets(string database) {
|
||
|
if (database != null && StringUtil.StringStartsWith(database, '[') && StringUtil.StringEndsWith(database, ']'))
|
||
|
return database.Substring(1, database.Length-2);
|
||
|
return database;
|
||
|
}
|
||
|
|
||
|
static void EnsureDatabaseExists(string database, SqlConnection sqlConnection) {
|
||
|
SqlCommand cmd;
|
||
|
string databaseNoSquareBrackets = RemoveSquareBrackets(database);
|
||
|
|
||
|
// First, make sure the database exists
|
||
|
cmd = new SqlCommand("SELECT DB_ID(@database)", sqlConnection);
|
||
|
cmd.Parameters.Add(new SqlParameter("@database", databaseNoSquareBrackets));
|
||
|
|
||
|
object res = cmd.ExecuteScalar();
|
||
|
|
||
|
if (res == null || res == System.DBNull.Value) {
|
||
|
// The database doesn't even exist.
|
||
|
throw new HttpException(
|
||
|
SR.GetString(SR.SQL_Services_Error_Cant_Uninstall_Nonexisting_Database,
|
||
|
databaseNoSquareBrackets));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Add/Remove all requested general features
|
||
|
static void SetupApplicationServices(
|
||
|
string server,
|
||
|
string user,
|
||
|
string password,
|
||
|
bool trusted,
|
||
|
string connectionString,
|
||
|
string database,
|
||
|
string dbFileName,
|
||
|
SqlFeatures features,
|
||
|
bool install )
|
||
|
{
|
||
|
SqlConnection sqlConnection = null;
|
||
|
ArrayList files;
|
||
|
|
||
|
Debug.Trace("SqlServices",
|
||
|
"SetupApplicationServices called: server=" + server + ", database=" +
|
||
|
database + ", user=" + user + ", password=" + password +
|
||
|
", trusted=" + trusted + ", connectionString=" + connectionString +
|
||
|
", features=" + features + ", install=" + install);
|
||
|
|
||
|
ApplicationServicesParamCheck(features, ref database);
|
||
|
|
||
|
files = GetFiles(install, features);
|
||
|
|
||
|
try {
|
||
|
sqlConnection = GetSqlConnection(server, user, password, trusted, connectionString);
|
||
|
|
||
|
// If uninstall, make sure all the asp.net tables are empty
|
||
|
if (!install) {
|
||
|
EnsureDatabaseExists(database, sqlConnection);
|
||
|
string databaseNoSquareBrackets = RemoveSquareBrackets(database);
|
||
|
if (sqlConnection.Database != databaseNoSquareBrackets)
|
||
|
sqlConnection.ChangeDatabase(databaseNoSquareBrackets);
|
||
|
int itablesToCheck = 0;
|
||
|
for (int i=0; i < s_featureInfos.Length; i++)
|
||
|
if (((int)s_featureInfos[i]._feature & (int)features) == (int)s_featureInfos[i]._feature)
|
||
|
itablesToCheck |= s_featureInfos[i]._dataCheckBitMask;
|
||
|
SqlCommand cmd = new SqlCommand("dbo.aspnet_AnyDataInTables", sqlConnection);
|
||
|
cmd.Parameters.Add(new SqlParameter("@TablesToCheck", itablesToCheck));
|
||
|
cmd.CommandType = CommandType.StoredProcedure;
|
||
|
string table = null;
|
||
|
try {
|
||
|
table = cmd.ExecuteScalar() as string;
|
||
|
} catch (SqlException e) {
|
||
|
if (e.Number != 2812)
|
||
|
throw;
|
||
|
}
|
||
|
if (!string.IsNullOrEmpty(table))
|
||
|
throw new NotSupportedException(
|
||
|
SR.GetString(SR.SQL_Services_Error_Cant_Uninstall_Nonempty_Table,
|
||
|
table, database));
|
||
|
}
|
||
|
|
||
|
// Load and run the sql file for each feature
|
||
|
foreach (string sqlFile in files) {
|
||
|
ExecuteFile(sqlFile, server, database, dbFileName, sqlConnection, false, false, SessionStateType.Temporary);
|
||
|
}
|
||
|
}
|
||
|
finally {
|
||
|
if (sqlConnection != null) {
|
||
|
try {
|
||
|
sqlConnection.Close();
|
||
|
}
|
||
|
catch {
|
||
|
}
|
||
|
finally {
|
||
|
sqlConnection = null;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
static void SessionStateParamCheck(SessionStateType type, ref string customDatabase) {
|
||
|
if (type == SessionStateType.Custom && String.IsNullOrEmpty(customDatabase)) {
|
||
|
throw new ArgumentException(
|
||
|
SR.GetString(SR.SQL_Services_Error_missing_custom_database), "customDatabase");
|
||
|
}
|
||
|
|
||
|
if (type != SessionStateType.Custom && customDatabase != null) {
|
||
|
throw new ArgumentException(
|
||
|
SR.GetString(SR.SQL_Services_Error_Cant_use_custom_database), "customDatabase");
|
||
|
}
|
||
|
CheckDatabaseName(ref customDatabase);
|
||
|
}
|
||
|
|
||
|
static void SetupSessionState(string server, string user, string password, bool trusted,
|
||
|
string connectionString, string customDatabase, SessionStateType type, bool install) {
|
||
|
SqlConnection sqlConnection = null;
|
||
|
|
||
|
Debug.Trace("SqlServices",
|
||
|
"SetupSessionState called: server=" + server + ", customDatabase=" +
|
||
|
customDatabase + ", user=" + user + ", password=" + password +
|
||
|
", trusted=" + trusted + ", connectionString=" + connectionString +
|
||
|
", type=" + type + ", install=" + install);
|
||
|
|
||
|
SessionStateParamCheck(type, ref customDatabase);
|
||
|
|
||
|
try {
|
||
|
sqlConnection = GetSqlConnection(server, user, password, trusted, connectionString);
|
||
|
|
||
|
if (!install && type == SessionStateType.Custom) {
|
||
|
EnsureDatabaseExists(customDatabase, sqlConnection);
|
||
|
}
|
||
|
|
||
|
// Load and run the sql file for each feature
|
||
|
ExecuteSessionFile(install ? SESSION_STATE_INSTALL_FILE : SESSION_STATE_UNINSTALL_FILE,
|
||
|
server, customDatabase, null, sqlConnection, install, type);
|
||
|
}
|
||
|
finally {
|
||
|
if (sqlConnection != null) {
|
||
|
try {
|
||
|
sqlConnection.Close();
|
||
|
}
|
||
|
catch {
|
||
|
}
|
||
|
finally {
|
||
|
sqlConnection = null;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
static string ConstructConnectionString(string server, string user, string password, bool trusted) {
|
||
|
string connectionString = null;
|
||
|
|
||
|
// Construct the connection string
|
||
|
|
||
|
if (String.IsNullOrEmpty(server)) {
|
||
|
throw ExceptionUtil.ParameterNullOrEmpty("server");
|
||
|
}
|
||
|
|
||
|
connectionString += "server=" + server;
|
||
|
|
||
|
if (trusted) {
|
||
|
connectionString += ";Trusted_Connection=true;";
|
||
|
}
|
||
|
else {
|
||
|
if (String.IsNullOrEmpty(user)) {
|
||
|
throw ExceptionUtil.ParameterNullOrEmpty("user");
|
||
|
}
|
||
|
|
||
|
connectionString += ";UID=" + user + ";" + "PWD=" + password + ";";
|
||
|
}
|
||
|
|
||
|
return connectionString;
|
||
|
|
||
|
}
|
||
|
|
||
|
static SqlConnection GetSqlConnection(string server, string user, string password,
|
||
|
bool trusted, string connectionString) {
|
||
|
SqlConnection sqlConnection;
|
||
|
|
||
|
if (connectionString == null) {
|
||
|
connectionString = ConstructConnectionString(server, user, password, trusted);
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
Debug.Trace("SqlServices", "Connecting to SQL: " + connectionString);
|
||
|
sqlConnection = new SqlConnection(connectionString);
|
||
|
sqlConnection.Open();
|
||
|
}
|
||
|
catch (Exception e) {
|
||
|
sqlConnection = null;
|
||
|
throw new HttpException(
|
||
|
SR.GetString(SR.SQL_Services_Cant_connect_sql_database),
|
||
|
e);
|
||
|
}
|
||
|
|
||
|
return sqlConnection;
|
||
|
}
|
||
|
}
|
||
|
}
|