Изменение данных соединения в файле PowerPivot - PullRequest
1 голос
/ 04 мая 2011

Итак, я столкнулся с интересной проблемой.Мне нужно указать пользовательское предложение WHERE в запросе PowerPivot.Я должен изменить это на основе внешних условий.Я хотел бы отредактировать файл и сохранить копию.есть идеи как это сделать?Я открыл файл PowerPivot из двоичного файла, но он выглядит зашифрованным ...

Ответы [ 2 ]

1 голос
/ 16 июля 2011

Вы можете перейти к существующим соединениям, а затем сделать там обновление.Если вы снова откроете тот же источник данных (SQL, SSRS или что-то еще) вместо изменения параметров существующего соединения, это замедлит работу, поскольку PowerPivot будет рассматривать их как отдельные соединения.

0 голосов
/ 04 ноября 2011

Решением было открыть книгу Excel в виде Zip (используя класс Package).

Если вы хотите изменить запросы, вы можете это сделать.Файл в /xl/customData/item1.data представляет собой файл резервной копии, представляющий базу данных PowerPivot (которая является просто базой данных служб Analysis Services, работающей в режиме Vertipaq), используемой для обработки запросов.Вам необходимо восстановить файл в экземпляр SSAS , работающий в режиме Vertipaq .Как только это будет сделано, создайте сценарий запросов как сценарий ALTER.Измените сценарии (в этом случае замените @projectId моим фактическим идентификатором проекта), затем запустите их для базы данных.Как только все это будет сделано, создайте резервную копию базы данных и поместите обратно в книгу Excel.Это изменяет запросы.

Данные соединения хранятся в файле /xl/connections.xml.Откройте это, измените и замените.Перепакуйте все это снова, и теперь у вас снова есть рабочая тетрадь.

Вот код, который я сделал.Вам придется вызывать методы так, как вам нужно.Основная идея есть, хотя ...

    const string DBName = "Testing";
    const string OriginalBackupPath = @"\\MyLocation\BKUP.abf";
    const string ModifiedBackupPath = @"\\MyLocation\BKUPAfter.abf";
    const string ServerPath = @"machineName\powerpivot";

    private static readonly Server srv = new Server();
    private static readonly Scripter scripter = new Scripter();
    private static Database db;

    private static byte[] GetPackagePartContents(string packagePath, string partPath)
    {
        var pack = Package.Open(packagePath, FileMode.OpenOrCreate, FileAccess.ReadWrite);
        var part = pack.GetPart(new Uri(partPath, UriKind.Relative));
        var stream = part.GetStream();
        var b = new byte[stream.Length];
        stream.Read(b, 0, b.Length);
        stream.Flush();
        stream.Close();
        pack.Flush();
        pack.Close();
        return b;
    }

    private static void WritePackagePartContents(string packagePath, string partPath, byte[] contents)
    {
        var uri = new Uri(partPath, UriKind.Relative);
        var pack = Package.Open(packagePath, FileMode.OpenOrCreate, FileAccess.ReadWrite);
        var part = pack.GetPart(uri);
        var type = part.ContentType;
        pack.DeletePart(uri);
        pack.CreatePart(uri, type);
        part = pack.GetPart(uri);
        var stream = part.GetStream();
        stream.Write(contents, 0, contents.Length);
        stream.Flush();
        stream.Close();
        pack.Flush();
        pack.Close();
    }

    private static void RestoreBackup(string server, string dbName, string backupPath)
    {
        srv.Connect(server);
        if (srv.Databases.FindByName(dbName) != null) { srv.Databases.FindByName(dbName).Drop(); srv.Update(); }
        srv.Restore(backupPath, dbName, true);
        srv.Update();
        srv.Refresh();
    }

    private static void WriteContentsToFile(byte[] contents, string filePath)
    {
        var fileStream = File.Open(filePath, FileMode.OpenOrCreate, FileAccess.Write);
        fileStream.Write(contents, 0, contents.Length);
        fileStream.Flush();
        fileStream.Close();
    }

    private static byte[] ReadContentsFromFile(string filePath)
    {
        var fileStream = File.Open(filePath, FileMode.OpenOrCreate, FileAccess.Write);
        var b = new byte[fileStream.Length];
        fileStream.Read(b, 0, b.Length);
        fileStream.Flush();
        fileStream.Close();
        return b;
    }

    private static XDocument GetAlterScript(MajorObject obj)
    {
        var stream = new MemoryStream();
        var streamWriter = XmlWriter.Create(stream);
        scripter.ScriptAlter(new[] { obj }, streamWriter, false);
        streamWriter.Flush();
        streamWriter.Close();
        stream.Flush();
        stream.Position = 0;
        var b = new byte[stream.Length];
        stream.Read(b, 0, b.Length);
        var alterString = new string(Encoding.UTF8.GetString(b).ToCharArray().Where(w => w != 65279).ToArray());
        var alter = XDocument.Parse(alterString);
        stream.Close();
        return alter;
    }

    private static void ExecuteScript(string script)
    {
        srv.Execute(script);
        srv.Update();
        db.Process();
        db.Refresh();
    }



    private static void ProcessPowerpointQueries(string bookUrl, string projectId)
    {
        byte[] b = GetPackagePartContents(bookUrl, "/xl/customData/item1.data");
        WriteContentsToFile(b, OriginalBackupPath);
        RestoreBackup(ServerPath, DBName, OriginalBackupPath);
        var db = srv.Databases.GetByName(DBName);
        var databaseView = db.DataSourceViews.FindByName("Sandbox");
        var databaseViewAlter = GetAlterScript(databaseView);
        var cube = db.Cubes.FindByName("Sandbox");
        var measureGroup = cube.MeasureGroups.FindByName("Query");
        var partition = measureGroup.Partitions.FindByName("Query");
        var partitionAlter = GetAlterScript(partition);
        var regex = new Regex(@"\s@projectid=\w*[ ,]");
        var newDatabaseViewAlter = databaseViewAlter.ToString().Replace(regex.Match(databaseViewAlter.ToString()).Value.Trim(',',' '), @"@projectid=" + projectId);
        ExecuteScript(newDatabaseViewAlter);
        var newPartitionAlter = partitionAlter.ToString().Replace(regex.Match(partitionAlter.ToString()).Value.Trim(',', ' '), @"@projectid=" + projectId);
        ExecuteScript(newPartitionAlter);
        db.Backup(ModifiedBackupPath, true);
        WritePackagePartContents(bookUrl, @"/xl/customData/item1.data", ReadContentsFromFile(ModifiedBackupPath));
        db.Drop();
        srv.Disconnect();
    }

    private static void ProcessWorkbookLinks(string bookUrl, string newCoreUrl)
    {
        var connectionsFile = GetPackagePartContents(bookUrl, @"/xl/connections.xml");
        var connectionsXml = Encoding.UTF8.GetString(connectionsFile);
        connectionsXml = connectionsXml.Replace(
            new Regex(@"Data Source=\S*;").Match(connectionsXml).Value.Trim(';'), @"Data Source=" + newCoreUrl);
        WritePackagePartContents(bookUrl, @"/xl/connections.xml", connectionsXml.Replace(@"https://server/site/", newCoreUrl).ToCharArray().Select(Convert.ToByte).ToArray());
    }
...