xref: /reactos/dll/win32/msi/update.c (revision 4e5e72fa)
1 /*
2  * Implementation of the Microsoft Installer (msi.dll)
3  *
4  * Copyright 2004 Mike McCormack for CodeWeavers
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20 
21 #include <stdarg.h>
22 
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winerror.h"
26 #include "wine/debug.h"
27 #include "msi.h"
28 #include "msiquery.h"
29 #include "objbase.h"
30 #include "objidl.h"
31 #include "msipriv.h"
32 #include "winnls.h"
33 
34 #include "query.h"
35 
36 WINE_DEFAULT_DEBUG_CHANNEL(msidb);
37 
38 
39 /* below is the query interface to a table */
40 
41 struct update_view
42 {
43     MSIVIEW          view;
44     MSIDATABASE     *db;
45     MSIVIEW         *wv;
46     column_info     *vals;
47 };
48 
49 static UINT UPDATE_fetch_int( struct tagMSIVIEW *view, UINT row, UINT col, UINT *val )
50 {
51     struct update_view *uv = (struct update_view *)view;
52 
53     TRACE("%p %d %d %p\n", uv, row, col, val );
54 
55     return ERROR_FUNCTION_FAILED;
56 }
57 
58 static UINT UPDATE_execute( struct tagMSIVIEW *view, MSIRECORD *record )
59 {
60     struct update_view *uv = (struct update_view *)view;
61     UINT i, r, col_count = 0, row_count = 0;
62     MSIRECORD *values = NULL;
63     MSIRECORD *where = NULL;
64     MSIVIEW *wv;
65     UINT cols_count, where_count;
66     column_info *col = uv->vals;
67 
68     TRACE("%p %p\n", uv, record );
69 
70     /* extract the where markers from the record */
71     if (record)
72     {
73         r = MSI_RecordGetFieldCount(record);
74 
75         for (i = 0; col; col = col->next)
76             i++;
77 
78         cols_count = i;
79         where_count = r - i;
80 
81         if (where_count > 0)
82         {
83             where = MSI_CreateRecord(where_count);
84 
85             if (where)
86                 for (i = 1; i <= where_count; i++)
87                     MSI_RecordCopyField(record, cols_count + i, where, i);
88         }
89     }
90 
91     wv = uv->wv;
92     if( !wv )
93     {
94         r = ERROR_FUNCTION_FAILED;
95         goto done;
96     }
97 
98     r = wv->ops->execute( wv, where );
99     TRACE("tv execute returned %x\n", r);
100     if( r )
101         goto done;
102 
103     r = wv->ops->get_dimensions( wv, &row_count, &col_count );
104     if( r )
105         goto done;
106 
107     values = msi_query_merge_record( col_count, uv->vals, record );
108     if (!values)
109     {
110         r = ERROR_FUNCTION_FAILED;
111         goto done;
112     }
113 
114     for ( i=0; i<row_count; i++ )
115     {
116         r = msi_select_update( wv, values, i );
117         if (r != ERROR_SUCCESS)
118             break;
119     }
120 
121 done:
122     if ( where ) msiobj_release( &where->hdr );
123     if ( values ) msiobj_release( &values->hdr );
124 
125     return r;
126 }
127 
128 
129 static UINT UPDATE_close( struct tagMSIVIEW *view )
130 {
131     struct update_view *uv = (struct update_view *)view;
132     MSIVIEW *wv;
133 
134     TRACE("%p\n", uv);
135 
136     wv = uv->wv;
137     if( !wv )
138         return ERROR_FUNCTION_FAILED;
139 
140     return wv->ops->close( wv );
141 }
142 
143 static UINT UPDATE_get_dimensions( struct tagMSIVIEW *view, UINT *rows, UINT *cols )
144 {
145     struct update_view *uv = (struct update_view *)view;
146     MSIVIEW *wv;
147 
148     TRACE("%p %p %p\n", uv, rows, cols );
149 
150     wv = uv->wv;
151     if( !wv )
152         return ERROR_FUNCTION_FAILED;
153 
154     return wv->ops->get_dimensions( wv, rows, cols );
155 }
156 
157 static UINT UPDATE_get_column_info( struct tagMSIVIEW *view, UINT n, LPCWSTR *name,
158                                     UINT *type, BOOL *temporary, LPCWSTR *table_name )
159 {
160     struct update_view *uv = (struct update_view *)view;
161     MSIVIEW *wv;
162 
163     TRACE("%p %d %p %p %p %p\n", uv, n, name, type, temporary, table_name );
164 
165     wv = uv->wv;
166     if( !wv )
167         return ERROR_FUNCTION_FAILED;
168 
169     return wv->ops->get_column_info( wv, n, name, type, temporary, table_name );
170 }
171 
172 static UINT UPDATE_modify( struct tagMSIVIEW *view, MSIMODIFY eModifyMode,
173                            MSIRECORD *rec, UINT row )
174 {
175     struct update_view *uv = (struct update_view *)view;
176 
177     TRACE("%p %d %p\n", uv, eModifyMode, rec );
178 
179     return ERROR_FUNCTION_FAILED;
180 }
181 
182 static UINT UPDATE_delete( struct tagMSIVIEW *view )
183 {
184     struct update_view *uv = (struct update_view *)view;
185     MSIVIEW *wv;
186 
187     TRACE("%p\n", uv );
188 
189     wv = uv->wv;
190     if( wv )
191         wv->ops->delete( wv );
192     msiobj_release( &uv->db->hdr );
193     free( uv );
194 
195     return ERROR_SUCCESS;
196 }
197 
198 static const MSIVIEWOPS update_ops =
199 {
200     UPDATE_fetch_int,
201     NULL,
202     NULL,
203     NULL,
204     NULL,
205     NULL,
206     NULL,
207     NULL,
208     UPDATE_execute,
209     UPDATE_close,
210     UPDATE_get_dimensions,
211     UPDATE_get_column_info,
212     UPDATE_modify,
213     UPDATE_delete,
214     NULL,
215     NULL,
216     NULL,
217     NULL,
218 };
219 
220 UINT UPDATE_CreateView( MSIDATABASE *db, MSIVIEW **view, LPWSTR table,
221                         column_info *columns, struct expr *expr )
222 {
223     struct update_view *uv = NULL;
224     UINT r;
225     MSIVIEW *sv = NULL, *wv = NULL;
226 
227     TRACE("%p\n", uv );
228 
229     if (expr)
230         r = WHERE_CreateView( db, &wv, table, expr );
231     else
232         r = TABLE_CreateView( db, table, &wv );
233 
234     if( r != ERROR_SUCCESS )
235         return r;
236 
237     /* then select the columns we want */
238     r = SELECT_CreateView( db, &sv, wv, columns );
239     if( r != ERROR_SUCCESS )
240     {
241         wv->ops->delete( wv );
242         return r;
243     }
244 
245     uv = calloc( 1, sizeof *uv );
246     if( !uv )
247     {
248         wv->ops->delete( wv );
249         return ERROR_FUNCTION_FAILED;
250     }
251 
252     /* fill the structure */
253     uv->view.ops = &update_ops;
254     msiobj_addref( &db->hdr );
255     uv->db = db;
256     uv->vals = columns;
257     uv->wv = sv;
258     *view = (MSIVIEW*) uv;
259 
260     return ERROR_SUCCESS;
261 }
262