diff options
| author | Rob Mensching <rob@firegiant.com> | 2017-10-14 16:12:07 -0700 |
|---|---|---|
| committer | Rob Mensching <rob@firegiant.com> | 2017-10-14 16:12:07 -0700 |
| commit | dbde9e7104b907bbbaea17e21247d8cafc8b3a4c (patch) | |
| tree | 0f5fbbb6fe12c6b2e5e622a0e18ce4c5b4eb2b96 /src/WixToolset.Core.WindowsInstaller/Msi/View.cs | |
| parent | fbf986eb97f68396797a89fc7d40dec07b775440 (diff) | |
| download | wix-dbde9e7104b907bbbaea17e21247d8cafc8b3a4c.tar.gz wix-dbde9e7104b907bbbaea17e21247d8cafc8b3a4c.tar.bz2 wix-dbde9e7104b907bbbaea17e21247d8cafc8b3a4c.zip | |
Massive refactoring to introduce the concept of IBackend
Diffstat (limited to 'src/WixToolset.Core.WindowsInstaller/Msi/View.cs')
| -rw-r--r-- | src/WixToolset.Core.WindowsInstaller/Msi/View.cs | 189 |
1 files changed, 189 insertions, 0 deletions
diff --git a/src/WixToolset.Core.WindowsInstaller/Msi/View.cs b/src/WixToolset.Core.WindowsInstaller/Msi/View.cs new file mode 100644 index 00000000..d6542824 --- /dev/null +++ b/src/WixToolset.Core.WindowsInstaller/Msi/View.cs | |||
| @@ -0,0 +1,189 @@ | |||
| 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.Msi | ||
| 4 | { | ||
| 5 | using System; | ||
| 6 | using System.ComponentModel; | ||
| 7 | using System.Globalization; | ||
| 8 | using WixToolset.Core.Native; | ||
| 9 | |||
| 10 | /// <summary> | ||
| 11 | /// Enumeration of different modify modes. | ||
| 12 | /// </summary> | ||
| 13 | public enum ModifyView | ||
| 14 | { | ||
| 15 | /// <summary> | ||
| 16 | /// Writes current data in the cursor to a table row. Updates record if the primary | ||
| 17 | /// keys match an existing row and inserts if they do not match. Fails with a read-only | ||
| 18 | /// database. This mode cannot be used with a view containing joins. | ||
| 19 | /// </summary> | ||
| 20 | Assign = MsiInterop.MSIMODIFYASSIGN, | ||
| 21 | |||
| 22 | /// <summary> | ||
| 23 | /// Remove a row from the table. You must first call the Fetch function with the same | ||
| 24 | /// record. Fails if the row has been deleted. Works only with read-write records. This | ||
| 25 | /// mode cannot be used with a view containing joins. | ||
| 26 | /// </summary> | ||
| 27 | Delete = MsiInterop.MSIMODIFYDELETE, | ||
| 28 | |||
| 29 | /// <summary> | ||
| 30 | /// Inserts a record. Fails if a row with the same primary keys exists. Fails with a read-only | ||
| 31 | /// database. This mode cannot be used with a view containing joins. | ||
| 32 | /// </summary> | ||
| 33 | Insert = MsiInterop.MSIMODIFYINSERT, | ||
| 34 | |||
| 35 | /// <summary> | ||
| 36 | /// Inserts a temporary record. The information is not persistent. Fails if a row with the | ||
| 37 | /// same primary key exists. Works only with read-write records. This mode cannot be | ||
| 38 | /// used with a view containing joins. | ||
| 39 | /// </summary> | ||
| 40 | InsertTemporary = MsiInterop.MSIMODIFYINSERTTEMPORARY, | ||
| 41 | |||
| 42 | /// <summary> | ||
| 43 | /// Inserts or validates a record in a table. Inserts if primary keys do not match any row | ||
| 44 | /// and validates if there is a match. Fails if the record does not match the data in | ||
| 45 | /// the table. Fails if there is a record with a duplicate key that is not identical. | ||
| 46 | /// Works only with read-write records. This mode cannot be used with a view containing joins. | ||
| 47 | /// </summary> | ||
| 48 | Merge = MsiInterop.MSIMODIFYMERGE, | ||
| 49 | |||
| 50 | /// <summary> | ||
| 51 | /// Refreshes the information in the record. Must first call Fetch with the | ||
| 52 | /// same record. Fails for a deleted row. Works with read-write and read-only records. | ||
| 53 | /// </summary> | ||
| 54 | Refresh = MsiInterop.MSIMODIFYREFRESH, | ||
| 55 | |||
| 56 | /// <summary> | ||
| 57 | /// Updates or deletes and inserts a record into a table. Must first call Fetch with | ||
| 58 | /// the same record. Updates record if the primary keys are unchanged. Deletes old row and | ||
| 59 | /// inserts new if primary keys have changed. Fails with a read-only database. This mode cannot | ||
| 60 | /// be used with a view containing joins. | ||
| 61 | /// </summary> | ||
| 62 | Replace = MsiInterop.MSIMODIFYREPLACE, | ||
| 63 | |||
| 64 | /// <summary> | ||
| 65 | /// Refreshes the information in the supplied record without changing the position in the | ||
| 66 | /// result set and without affecting subsequent fetch operations. The record may then | ||
| 67 | /// be used for subsequent Update, Delete, and Refresh. All primary key columns of the | ||
| 68 | /// table must be in the query and the record must have at least as many fields as the | ||
| 69 | /// query. Seek cannot be used with multi-table queries. This mode cannot be used with | ||
| 70 | /// a view containing joins. See also the remarks. | ||
| 71 | /// </summary> | ||
| 72 | Seek = MsiInterop.MSIMODIFYSEEK, | ||
| 73 | |||
| 74 | /// <summary> | ||
| 75 | /// Updates an existing record. Non-primary keys only. Must first call Fetch. Fails with a | ||
| 76 | /// deleted record. Works only with read-write records. | ||
| 77 | /// </summary> | ||
| 78 | Update = MsiInterop.MSIMODIFYUPDATE | ||
| 79 | } | ||
| 80 | |||
| 81 | /// <summary> | ||
| 82 | /// Wrapper class for MSI API views. | ||
| 83 | /// </summary> | ||
| 84 | internal sealed class View : MsiHandle | ||
| 85 | { | ||
| 86 | /// <summary> | ||
| 87 | /// Constructor that creates a view given a database handle and a query. | ||
| 88 | /// </summary> | ||
| 89 | /// <param name="db">Handle to the database to run the query on.</param> | ||
| 90 | /// <param name="query">Query to be executed.</param> | ||
| 91 | public View(Database db, string query) | ||
| 92 | { | ||
| 93 | if (null == db) | ||
| 94 | { | ||
| 95 | throw new ArgumentNullException("db"); | ||
| 96 | } | ||
| 97 | |||
| 98 | if (null == query) | ||
| 99 | { | ||
| 100 | throw new ArgumentNullException("query"); | ||
| 101 | } | ||
| 102 | |||
| 103 | uint handle = 0; | ||
| 104 | |||
| 105 | int error = MsiInterop.MsiDatabaseOpenView(db.Handle, query, out handle); | ||
| 106 | if (0 != error) | ||
| 107 | { | ||
| 108 | throw new MsiException(error); | ||
| 109 | } | ||
| 110 | |||
| 111 | this.Handle = handle; | ||
| 112 | } | ||
| 113 | |||
| 114 | /// <summary> | ||
| 115 | /// Executes a view with no customizable parameters. | ||
| 116 | /// </summary> | ||
| 117 | public void Execute() | ||
| 118 | { | ||
| 119 | this.Execute(null); | ||
| 120 | } | ||
| 121 | |||
| 122 | /// <summary> | ||
| 123 | /// Executes a query substituing the values from the records into the customizable parameters | ||
| 124 | /// in the view. | ||
| 125 | /// </summary> | ||
| 126 | /// <param name="record">Record containing parameters to be substituded into the view.</param> | ||
| 127 | public void Execute(Record record) | ||
| 128 | { | ||
| 129 | int error = MsiInterop.MsiViewExecute(this.Handle, null == record ? 0 : record.Handle); | ||
| 130 | if (0 != error) | ||
| 131 | { | ||
| 132 | throw new MsiException(error); | ||
| 133 | } | ||
| 134 | } | ||
| 135 | |||
| 136 | /// <summary> | ||
| 137 | /// Fetches the next row in the view. | ||
| 138 | /// </summary> | ||
| 139 | /// <returns>Returns the fetched record; otherwise null.</returns> | ||
| 140 | public Record Fetch() | ||
| 141 | { | ||
| 142 | uint recordHandle; | ||
| 143 | |||
| 144 | int error = MsiInterop.MsiViewFetch(this.Handle, out recordHandle); | ||
| 145 | if (259 == error) | ||
| 146 | { | ||
| 147 | return null; | ||
| 148 | } | ||
| 149 | else if (0 != error) | ||
| 150 | { | ||
| 151 | throw new MsiException(error); | ||
| 152 | } | ||
| 153 | |||
| 154 | return new Record(recordHandle); | ||
| 155 | } | ||
| 156 | |||
| 157 | /// <summary> | ||
| 158 | /// Updates a fetched record. | ||
| 159 | /// </summary> | ||
| 160 | /// <param name="type">Type of modification mode.</param> | ||
| 161 | /// <param name="record">Record to be modified.</param> | ||
| 162 | public void Modify(ModifyView type, Record record) | ||
| 163 | { | ||
| 164 | int error = MsiInterop.MsiViewModify(this.Handle, Convert.ToInt32(type, CultureInfo.InvariantCulture), record.Handle); | ||
| 165 | if (0 != error) | ||
| 166 | { | ||
| 167 | throw new MsiException(error); | ||
| 168 | } | ||
| 169 | } | ||
| 170 | |||
| 171 | /// <summary> | ||
| 172 | /// Returns a record containing column names or definitions. | ||
| 173 | /// </summary> | ||
| 174 | /// <param name="columnType">Specifies a flag indicating what type of information is needed. Either MSICOLINFO_NAMES or MSICOLINFO_TYPES.</param> | ||
| 175 | /// <returns>The record containing information about the column.</returns> | ||
| 176 | public Record GetColumnInfo(int columnType) | ||
| 177 | { | ||
| 178 | uint recordHandle; | ||
| 179 | |||
| 180 | int error = MsiInterop.MsiViewGetColumnInfo(this.Handle, columnType, out recordHandle); | ||
| 181 | if (0 != error) | ||
| 182 | { | ||
| 183 | throw new MsiException(error); | ||
| 184 | } | ||
| 185 | |||
| 186 | return new Record(recordHandle); | ||
| 187 | } | ||
| 188 | } | ||
| 189 | } | ||
