diff options
Diffstat (limited to 'src/dtf/WixToolset.Dtf.WindowsInstaller.Linq/QDatabase.cs')
| -rw-r--r-- | src/dtf/WixToolset.Dtf.WindowsInstaller.Linq/QDatabase.cs | 214 |
1 files changed, 214 insertions, 0 deletions
diff --git a/src/dtf/WixToolset.Dtf.WindowsInstaller.Linq/QDatabase.cs b/src/dtf/WixToolset.Dtf.WindowsInstaller.Linq/QDatabase.cs new file mode 100644 index 00000000..b4de2f60 --- /dev/null +++ b/src/dtf/WixToolset.Dtf.WindowsInstaller.Linq/QDatabase.cs | |||
| @@ -0,0 +1,214 @@ | |||
| 1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
| 2 | |||
| 3 | namespace WixToolset.Dtf.WindowsInstaller.Linq | ||
| 4 | { | ||
| 5 | using System; | ||
| 6 | using System.IO; | ||
| 7 | using WixToolset.Dtf.WindowsInstaller.Linq.Entities; | ||
| 8 | |||
| 9 | /// <summary> | ||
| 10 | /// Allows any Database instance to be converted into a queryable database. | ||
| 11 | /// </summary> | ||
| 12 | public static class Queryable | ||
| 13 | { | ||
| 14 | /// <summary> | ||
| 15 | /// Converts any Database instance into a queryable database. | ||
| 16 | /// </summary> | ||
| 17 | /// <param name="db"></param> | ||
| 18 | /// <returns>Queryable database instance that operates on the same | ||
| 19 | /// MSI handle.</returns> | ||
| 20 | /// <remarks> | ||
| 21 | /// This extension method is meant for convenient on-the-fly conversion. | ||
| 22 | /// If the existing database instance already happens to be a QDatabase, | ||
| 23 | /// then it is returned unchanged. Otherwise since the new database | ||
| 24 | /// carries the same MSI handle, only one of the instances needs to be | ||
| 25 | /// closed, not both. | ||
| 26 | /// </remarks> | ||
| 27 | public static QDatabase AsQueryable(this Database db) | ||
| 28 | { | ||
| 29 | QDatabase qdb = db as QDatabase; | ||
| 30 | if (qdb == null && db != null) | ||
| 31 | { | ||
| 32 | qdb = new QDatabase(db.Handle, true, db.FilePath, db.OpenMode); | ||
| 33 | } | ||
| 34 | return qdb; | ||
| 35 | } | ||
| 36 | } | ||
| 37 | |||
| 38 | /// <summary> | ||
| 39 | /// Queryable MSI database - extends the base Database class with | ||
| 40 | /// LINQ query functionality along with predefined entity types | ||
| 41 | /// for common tables. | ||
| 42 | /// </summary> | ||
| 43 | public class QDatabase : Database | ||
| 44 | { | ||
| 45 | /// <summary> | ||
| 46 | /// Opens an existing database in read-only mode. | ||
| 47 | /// </summary> | ||
| 48 | /// <param name="filePath">Path to the database file.</param> | ||
| 49 | /// <exception cref="InstallerException">the database could not be created/opened</exception> | ||
| 50 | /// <remarks> | ||
| 51 | /// Because this constructor initiates database access, it cannot be used with a | ||
| 52 | /// running installation. | ||
| 53 | /// <para>The Database object should be <see cref="InstallerHandle.Close"/>d after use. | ||
| 54 | /// The finalizer will close the handle if it is still open, however due to the nondeterministic | ||
| 55 | /// nature of finalization it is best that the handle be closed manually as soon as it is no | ||
| 56 | /// longer needed, as leaving lots of unused handles open can degrade performance.</para> | ||
| 57 | /// </remarks> | ||
| 58 | public QDatabase(string filePath) | ||
| 59 | : base(filePath) | ||
| 60 | { | ||
| 61 | } | ||
| 62 | |||
| 63 | /// <summary> | ||
| 64 | /// Opens an existing database with another database as output. | ||
| 65 | /// </summary> | ||
| 66 | /// <param name="filePath">Path to the database to be read.</param> | ||
| 67 | /// <param name="outputPath">Open mode for the database</param> | ||
| 68 | /// <returns>Database object representing the created or opened database</returns> | ||
| 69 | /// <exception cref="InstallerException">the database could not be created/opened</exception> | ||
| 70 | /// <remarks> | ||
| 71 | /// When a database is opened as the output of another database, the summary information stream | ||
| 72 | /// of the output database is actually a read-only mirror of the original database and thus cannot | ||
| 73 | /// be changed. Additionally, it is not persisted with the database. To create or modify the | ||
| 74 | /// summary information for the output database it must be closed and re-opened. | ||
| 75 | /// <para>The returned Database object should be <see cref="InstallerHandle.Close"/>d after use. | ||
| 76 | /// The finalizer will close the handle if it is still open, however due to the nondeterministic | ||
| 77 | /// nature of finalization it is best that the handle be closed manually as soon as it is no | ||
| 78 | /// longer needed, as leaving lots of unused handles open can degrade performance.</para> | ||
| 79 | /// </remarks> | ||
| 80 | public QDatabase(string filePath, string outputPath) | ||
| 81 | : base(filePath, outputPath) | ||
| 82 | { | ||
| 83 | } | ||
| 84 | |||
| 85 | /// <summary> | ||
| 86 | /// Opens an existing database or creates a new one. | ||
| 87 | /// </summary> | ||
| 88 | /// <param name="filePath">Path to the database file. If an empty string | ||
| 89 | /// is supplied, a temporary database is created that is not persisted.</param> | ||
| 90 | /// <param name="mode">Open mode for the database</param> | ||
| 91 | /// <exception cref="InstallerException">the database could not be created/opened</exception> | ||
| 92 | /// <remarks> | ||
| 93 | /// To make and save changes to a database first open the database in transaction, | ||
| 94 | /// create or, or direct mode. After making the changes, always call the Commit method | ||
| 95 | /// before closing the database handle. The Commit method flushes all buffers. | ||
| 96 | /// <para>Always call the Commit method on a database that has been opened in direct | ||
| 97 | /// mode before closing the database. Failure to do this may corrupt the database.</para> | ||
| 98 | /// <para>Because this constructor initiates database access, it cannot be used with a | ||
| 99 | /// running installation.</para> | ||
| 100 | /// <para>The Database object should be <see cref="InstallerHandle.Close"/>d after use. | ||
| 101 | /// The finalizer will close the handle if it is still open, however due to the nondeterministic | ||
| 102 | /// nature of finalization it is best that the handle be closed manually as soon as it is no | ||
| 103 | /// longer needed, as leaving lots of unused handles open can degrade performance.</para> | ||
| 104 | /// </remarks> | ||
| 105 | public QDatabase(string filePath, DatabaseOpenMode mode) | ||
| 106 | : base(filePath, mode) | ||
| 107 | { | ||
| 108 | } | ||
| 109 | |||
| 110 | /// <summary> | ||
| 111 | /// Creates a new database from an MSI handle. | ||
| 112 | /// </summary> | ||
| 113 | /// <param name="handle">Native MSI database handle.</param> | ||
| 114 | /// <param name="ownsHandle">True if the handle should be closed | ||
| 115 | /// when the database object is disposed</param> | ||
| 116 | /// <param name="filePath">Path of the database file, if known</param> | ||
| 117 | /// <param name="openMode">Mode the handle was originally opened in</param> | ||
| 118 | protected internal QDatabase( | ||
| 119 | IntPtr handle, bool ownsHandle, string filePath, DatabaseOpenMode openMode) | ||
| 120 | : base(handle, ownsHandle, filePath, openMode) | ||
| 121 | { | ||
| 122 | } | ||
| 123 | |||
| 124 | /// <summary> | ||
| 125 | /// Gets or sets a log where all MSI SQL queries are written. | ||
| 126 | /// </summary> | ||
| 127 | /// <remarks> | ||
| 128 | /// The log can be useful for debugging, or simply to watch the LINQ magic in action. | ||
| 129 | /// </remarks> | ||
| 130 | public TextWriter Log { get; set; } | ||
| 131 | |||
| 132 | /// <summary> | ||
| 133 | /// Gets a queryable table from the datbaase. | ||
| 134 | /// </summary> | ||
| 135 | /// <param name="table">name of the table</param> | ||
| 136 | public QTable<QRecord> this[string table] | ||
| 137 | { | ||
| 138 | get | ||
| 139 | { | ||
| 140 | return new QTable<QRecord>(this, table); | ||
| 141 | } | ||
| 142 | } | ||
| 143 | |||
| 144 | #if !CODE_ANALYSIS | ||
| 145 | #region Queryable tables | ||
| 146 | |||
| 147 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 148 | public QTable<Component_> Components | ||
| 149 | { get { return new QTable<Component_>(this); } } | ||
| 150 | |||
| 151 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 152 | public QTable<CreateFolder_> CreateFolders | ||
| 153 | { get { return new QTable<CreateFolder_>(this); } } | ||
| 154 | |||
| 155 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 156 | public QTable<CustomAction_> CustomActions | ||
| 157 | { get { return new QTable<CustomAction_>(this); } } | ||
| 158 | |||
| 159 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 160 | public QTable<Directory_> Directories | ||
| 161 | { get { return new QTable<Directory_>(this); } } | ||
| 162 | |||
| 163 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 164 | public QTable<DuplicateFile_> DuplicateFiles | ||
| 165 | { get { return new QTable<DuplicateFile_>(this); } } | ||
| 166 | |||
| 167 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 168 | public QTable<Feature_> Features | ||
| 169 | { get { return new QTable<Feature_>(this); } } | ||
| 170 | |||
| 171 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 172 | public QTable<FeatureComponent_> FeatureComponents | ||
| 173 | { get { return new QTable<FeatureComponent_>(this); } } | ||
| 174 | |||
| 175 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 176 | public QTable<File_> Files | ||
| 177 | { get { return new QTable<File_>(this); } } | ||
| 178 | |||
| 179 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 180 | public QTable<FileHash_> FileHashes | ||
| 181 | { get { return new QTable<FileHash_>(this); } } | ||
| 182 | |||
| 183 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 184 | public QTable<InstallSequence_> InstallExecuteSequences | ||
| 185 | { get { return new QTable<InstallSequence_>(this, "InstallExecuteSequence"); } } | ||
| 186 | |||
| 187 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 188 | public QTable<InstallSequence_> InstallUISequences | ||
| 189 | { get { return new QTable<InstallSequence_>(this, "InstallUISequence"); } } | ||
| 190 | |||
| 191 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 192 | public QTable<LaunchCondition_> LaunchConditions | ||
| 193 | { get { return new QTable<LaunchCondition_>(this); } } | ||
| 194 | |||
| 195 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 196 | public QTable<Media_> Medias | ||
| 197 | { get { return new QTable<Media_>(this); } } | ||
| 198 | |||
| 199 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 200 | public QTable<Property_> Properties | ||
| 201 | { get { return new QTable<Property_>(this); } } | ||
| 202 | |||
| 203 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 204 | public QTable<Registry_> Registries | ||
| 205 | { get { return new QTable<Registry_>(this); } } | ||
| 206 | |||
| 207 | /// <summary>Queryable standard table with predefined specialized record type.</summary> | ||
| 208 | public QTable<RemoveFile_> RemoveFiles | ||
| 209 | { get { return new QTable<RemoveFile_>(this); } } | ||
| 210 | |||
| 211 | #endregion // Queryable tables | ||
| 212 | #endif // !CODE_ANALYSIS | ||
| 213 | } | ||
| 214 | } | ||
