С # SQL Server резервное копирование удаленной базы данных в удаленное место резервного копирования по умолчанию без прямого доступа к удаленному расположению?
TL;DR - я хочу, чтобы сервер делал резервную копию, а не мое приложение, потому что сервер настроен для этого, и у моего приложения не будет доступа.
Задний план
Моя компания создала программное обеспечение для клиентов 20 лет назад, написанное на Delphi 7/Pascal. Я переписываю программу на C#. В рамках переписывания я создал новые базы данных Firebird, Oracle и SQL Server. Федеральное регулирование требует, чтобы все существующие данные сохранялись, поэтому я создал инструмент модификации / преобразования базы данных, чтобы перейти от старой структуры базы данных к новой.
Прежде чем я начну вносить изменения, мне нужно сделать резервную копию существующей базы данных. Технический специалист, который будет запускать этот инструмент, не имеет доступа к своей локальной файловой структуре и ручного доступа к удаленному серверу, на котором размещена база данных. Инструмент обращается к зашифрованному.ini
-подобный файл в локальной системе, чтобы проанализировать компоненты строки подключения и создать объект подключения. Затем я использую этот объект подключения для подключения к той же базе данных, к которой настроен компьютер технического специалиста. В этой части все работает
Если я оставлю только путь резервного копирования по умолчанию, он попытается выполнить резервное копирование по пути по умолчанию, но на локальном компьютере (у технических специалистов нет доступа для создания, и мы не хотим, чтобы технический специалист имел доступ к.bak в любом случае). по умолчанию резервный путь - это сетевой путь, взятый из строки подключения, я получаю
SmoException: System.Data.SqlClient.SqlError: не удается открыть устройство резервного копирования. Ошибка операционной системы 67(не удается найти сетевое имя).
потому что путь к файлу не является сетевым ресурсом (и не будет), а учетные данные пользователя базы данных не могут получить доступ к этому пути извне SQL Server.
Возникает вопрос: как сделать резервную копию удаленного пути по умолчанию, как если бы я был на сервере?
Вот код, который генерирует ошибку выше (нулевой случай для удаленного).
public static void FullSqlBackup (Connection oldProactiveSql)
{
String sqlServerLogin = oldProactiveSql.UserName;
String password = oldProactiveSql.PassWord;
String instanceName = oldProactiveSql.InstanceName;
String remoteSvrName = oldProactiveSql.Ip + "," + oldProactiveSql.Port;
Server srv2;
Server srv3;
string device;
switch (oldProactiveSql.InstanceName)
{
case null:
ServerConnection srvConn2 = new ServerConnection(remoteSvrName);
srvConn2.LoginSecure = false;
srvConn2.Login = sqlServerLogin;
srvConn2.Password = password;
srv3 = new Server(srvConn2);
srv2 = null;
Console.WriteLine(srv3.Information.Version);
if (srv3.Settings.DefaultFile is null)
{
device = srv3.Information.RootDirectory + "\\DATA\\";
device = device.Substring(2);
device = oldProactiveSql.Ip + device;
}
else device = srv3.Settings.DefaultFile;
device = device.Substring(2);
device = string.Concat("\\\\", oldProactiveSql.Ip, device);
break;
default:
ServerConnection srvConn = new ServerConnection();
srvConn.ServerInstance = @".\" + instanceName;
srvConn.LoginSecure = false;
srvConn.Login = sqlServerLogin;
srvConn.Password = password;
srv2 = new Server(srvConn);
srv3 = null;
Console.WriteLine(srv2.Information.Version);
if (srv2.Settings.DefaultFile is null)
{
device = srv2.Information.RootDirectory + "\\DATA\\";
}
else device = srv2.Settings.DefaultFile;
break;
}
Backup bkpDbFull = new Backup();
bkpDbFull.Action = BackupActionType.Database;
bkpDbFull.Database = oldProactiveSql.DbName;
bkpDbFull.Devices.AddDevice(device, DeviceType.File);
bkpDbFull.BackupSetName = oldProactiveSql.DbName + " database Backup";
bkpDbFull.BackupSetDescription = oldProactiveSql.DbName + " database - Full Backup";
bkpDbFull.Initialize = true;
bkpDbFull.PercentComplete += CompletionStatusInPercent;
bkpDbFull.Complete += Backup_Completed;
switch (oldProactiveSql.InstanceName)
{
case null:
try
{
bkpDbFull.SqlBackup(srv3);
}
catch (Exception e)
{
Console.WriteLine (e);
Console.WriteLine(e.InnerException.Message);
throw;
}
break;
default:
try
{
bkpDbFull.SqlBackup(srv2);
}
catch (Exception e)
{
Console.WriteLine(e);
Console.WriteLine(e.InnerException.Message);
throw;
}
break;
}
}
Любая помощь будет оценена по достоинству, так как сейчас я просто бегаю по кругу.
Из комментариев ниже я попробую: 1. Динамически создать хранимую процедуру [BackupToDefault] в базе данных, а затем запустить ее. 2. Если это не удается, свяжите базу данных с собой. 3. Попробуйте - Exec [BackupToDefault] на [LinkedSelfSynonmym]
Пожелайте мне удачи, хотя это кажется запутанным и долгим, я надеюсь, что это сработает.
2 ответа
Спасибо @SeanLange
Совершенно уверен, что в этом случае вам нужно будет использовать динамический sql. Тогда путь будет относительно того, где выполняется sql.
Я изменил свой код, чтобы добавить:
private static void WriteBackupSp (Server remoteServer, string dbName, out string storedProcedure)
{
var s = "CREATE PROCEDURE [dbo].[ProactiveDBBackup]\n";
s += "AS\n";
s += "BEGIN\n";
s += "SET NOCOUNT ON\n";
s += "BACKUP DATABASE " + dbName + " TO DISK = \'" + string.Concat (remoteServer.BackupDirectory,@"\", dbName, ".bak") + "\'\n";
s += "END\n";
storedProcedure = s;
}
затем изменил последний переключатель на:
switch (oldProactiveSql.InstanceName)
{
case null:
try
{
WriteBackupSp (srv3, oldProactiveSql.DbName, out var storedProcedure);
ConnectionToolsUtility.GenerateSqlConnectionString (oldProactiveSql, out var cs);
using (SqlConnection connection = new SqlConnection (cs))
{
using (SqlCommand command = new SqlCommand (storedProcedure, connection))
{
connection.Open ();
command.ExecuteNonQuery ();
connection.Close ();
}
}
var execBackup = "EXEC [dbo].[ProactiveDBBackup]\n";
using (SqlConnection connection = new SqlConnection (cs))
{
using (SqlCommand command = new SqlCommand (execBackup, connection))
{
connection.Open ();
command.ExecuteNonQuery ();
connection.Close ();
}
}
}
catch (Exception e)
{
Console.WriteLine(e);
Console.WriteLine(e.InnerException.Message);
throw;
}
break;
default:
try { bkpDbFull.SqlBackup(srv2); }
catch (Exception e)
{
Console.WriteLine(e);
Console.WriteLine(e.InnerException.Message);
throw;
}
break;
}
И это позволило мне сделать резервную копию базы данных через строку подключения в хранилище резервных копий по умолчанию, не имея учетных данных с доступом к местоположению сетевого пути.
Для вдохновения... резервная копия разделена на 3 файла, каждый из которых находится в другом каталоге (каталог резервного копирования экземпляра sql и каталог по умолчанию для экземпляра sql и основной каталог базы данных)
//// compile with:
// /r:Microsoft.SqlServer.Smo.dll
// /r:Microsoft.SqlServer.SmoExtended.dll
// /r:Microsoft.SqlServer.ConnectionInfo.dll
using System;
using System.Data;
using Microsoft.SqlServer.Management.Smo;
using Microsoft.SqlServer.Management.Common;
namespace SMObackup
{
class Program
{
static void Main()
{
// For remote connection, remote server name / ServerInstance needs to be specified
ServerConnection srvConn2 = new ServerConnection("machinename"/* <--default sql instance on machinename*/); // or (@"machinename\sqlinstance") for named instances
srvConn2.LoginSecure = false;
srvConn2.Login = "smologin";
srvConn2.Password = "SmoL@gin11";
srvConn2.DatabaseName = "msdb";
Server srv3 = new Server(srvConn2);
//server info
Console.WriteLine("servername:{0} ---- version:{1}", srv3.Name, srv3.Information.Version);
//server root directory
string serverRootDir = srv3.Information.RootDirectory;
//server backup directory
string serverBackupDir = srv3.Settings.BackupDirectory;
//database primary directory
string databasePrimaryFilepath = srv3.Databases[srvConn2.DatabaseName].PrimaryFilePath;
Console.WriteLine("server_root_dir:{0}\nserver_backup_dir:{1}\ndatabase_primary_dir{2}", serverRootDir, serverBackupDir, databasePrimaryFilepath);
Backup bkpDbFull = new Backup();
bkpDbFull.Action = BackupActionType.Database;
//comment out copyonly ....
bkpDbFull.CopyOnly = true; //copy only, just for testing....avoid messing up with existing backup processes
bkpDbFull.Database = srvConn2.DatabaseName;
//backup file name
string backupfile = $"\\backuptest_{DateTime.Now.ToString("dd/MM/yyyy/hh/mm/ss")}.bak";
//add multiple files, in each location
bkpDbFull.Devices.AddDevice(serverRootDir + backupfile, DeviceType.File);
bkpDbFull.Devices.AddDevice(serverBackupDir + backupfile, DeviceType.File);
bkpDbFull.Devices.AddDevice(databasePrimaryFilepath + backupfile, DeviceType.File);
bkpDbFull.Initialize = true;
foreach (BackupDeviceItem backupdevice in bkpDbFull.Devices)
{
Console.WriteLine("deviceitem:{0}", backupdevice.Name);
}
//backup is split/divided amongst the 3 devices
bkpDbFull.SqlBackup(srv3);
Restore restore = new Restore();
restore.Devices.AddRange(bkpDbFull.Devices);
DataTable backupHeader = restore.ReadBackupHeader(srv3);
//IsCopyOnly=True
for (int r = 0; r < backupHeader.Rows.Count; r++)
{
for (int c = 0; c < backupHeader.Columns.Count; c++)
{
Console.Write("{0}={1}\n", backupHeader.Columns[c].ColumnName, (string.IsNullOrEmpty(backupHeader.Rows[r].ItemArray[c].ToString())? "**": backupHeader.Rows[r].ItemArray[c].ToString()) );
}
}
srvConn2.Disconnect(); //redundant
}
}
}