dbutils/object.c

Mon, 16 Dec 2024 18:38:11 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Mon, 16 Dec 2024 18:38:11 +0100
changeset 9
5785a693834c
parent 8
bd08116b8af4
child 10
80f9d007cb52
permissions
-rw-r--r--

prepare for dense queries

3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
1 /*
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
3 *
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
4 * Copyright 2024 Olaf Wintermann. All rights reserved.
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
5 *
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
6 * Redistribution and use in source and binary forms, with or without
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
7 * modification, are permitted provided that the following conditions are met:
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
8 *
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
9 * 1. Redistributions of source code must retain the above copyright
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
10 * notice, this list of conditions and the following disclaimer.
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
11 *
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
12 * 2. Redistributions in binary form must reproduce the above copyright
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
13 * notice, this list of conditions and the following disclaimer in the
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
14 * documentation and/or other materials provided with the distribution.
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
15 *
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
26 * POSSIBILITY OF SUCH DAMAGE.
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
27 */
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
28
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
29 #include <stdio.h>
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
30 #include <stdlib.h>
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
31 #include <string.h>
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
32
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
33 #include <cx/array_list.h>
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
34 #include <cx/linked_list.h>
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
35 #include <cx/hash_map.h>
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
36 #include <cx/printf.h>
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
37
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
38 #include "object.h"
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
39
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
40
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
41
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
42
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
43
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
44 DBUObjectBuilder* dbuObjectBuilder(DBUClass *type, DBUQuery *query, const CxAllocator *a) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
45 CxList *additionalQueries = cxLinkedListCreateSimple(sizeof(DBUBuilderQuery));
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
46 if(!additionalQueries) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
47 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
48 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
49 CxMap *subQueries = cxHashMapCreateSimple(CX_STORE_POINTERS);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
50 if(!subQueries) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
51 cxListDestroy(additionalQueries);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
52 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
53 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
54
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
55 DBUObjectBuilder *builder = malloc(sizeof(DBUObjectBuilder));
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
56 if(!builder) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
57 cxListDestroy(additionalQueries);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
58 cxMapDestroy(subQueries);
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
59 return NULL;
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
60 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
61
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
62 memset(builder, 0, sizeof(DBUObjectBuilder));
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
63 builder->allocator = a;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
64 builder->ctx = type->context;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
65 builder->mainQuery = query;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
66 builder->resultType = type;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
67 builder->additionalQueries = additionalQueries;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
68
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
69 return builder;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
70 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
71
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
72 void dbuObjectBuilderSetDenseResult(DBUObjectBuilder *builder, bool dense) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
73 builder->denseResult = dense;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
74 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
75
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
76 int dbuObjectBuilderAddSubquery(DBUObjectBuilder *builder, DBUClass *type, DBUQuery *subquery) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
77 return cxMapPut(builder->subQueries, type->name, subquery);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
78 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
79
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
80 int dbuObjectBuilderAddAdditionalQuery(DBUObjectBuilder *builder, DBUClass *type, DBUQuery *query) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
81 return cxListAdd(builder->additionalQueries, &(DBUBuilderQuery){ type, query });
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
82 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
83
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
84
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
85
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
86
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
87
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
88 // builder result constructors
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
89
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
90 static DBUObject result_create_obj(DBUObjectResult *result, DBUClass *type, const CxAllocator *a) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
91 void *obj = cxMalloc(a, type->obj_size);
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
92 if(obj) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
93 memset(obj, 0, type->obj_size);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
94 }
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
95 return obj;
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
96 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
97
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
98 static DBUObject valuelist_result_create(DBUObjectResult *result, DBUClass *type, const CxAllocator *a) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
99 memset(result->userdata2, 0, result->int1);
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
100 return result->userdata2;
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
101 }
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
102
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
103 static int list_result_add(DBUObjectResult *result, DBUObject parent, DBUClass *type, void *obj, CxList *fk, const CxAllocator *a) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
104 return cxListAdd(result->userdata1, obj);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
105 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
106
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
107 CxList* dbuObjectBuilderGetList(DBUObjectBuilder *builder) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
108 CxList *result_list = cxArrayListCreate(builder->allocator, NULL, CX_STORE_POINTERS, 8);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
109 if(!result_list) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
110 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
111 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
112 // TODO: the class needs a obj destructor func, because we need to be able
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
113 // to destroy partialy created lists
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
114 DBUObjectResult result = {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
115 .userdata1 = result_list,
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
116 .userdata2 = NULL,
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
117 .int1 = CX_STORE_POINTERS,
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
118 .create = result_create_obj,
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
119 .add = list_result_add
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
120 };
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
121 if(dbuObjectExec(builder, &result)) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
122 cxListDestroy(result_list);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
123 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
124 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
125 return result_list;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
126 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
127
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
128
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
129 CxList* dbuObjectBuilderGetValueList(DBUObjectBuilder *builder) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
130 CxList *result_list = cxArrayListCreate(builder->allocator, NULL, builder->resultType->obj_size, 8);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
131 if(!result_list) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
132 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
133 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
134 // TODO: the class needs a obj destructor func, because we need to be able
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
135 // to destroy partialy created lists
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
136 DBUObjectResult result = {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
137 .userdata1 = result_list,
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
138 .userdata2 = malloc(builder->resultType->obj_size),
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
139 .int1 = builder->resultType->obj_size,
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
140 .create = valuelist_result_create,
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
141 .add = list_result_add
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
142 };
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
143 if(!result.userdata2) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
144 cxListDestroy(result_list);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
145 return NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
146 }
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
147 if(dbuObjectExec(builder, &result)) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
148 cxListDestroy(result_list);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
149 return NULL;
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
150 }
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
151 return result_list;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
152 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
153
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
154 int dbuObjectBuilderGetArray(DBUObjectBuilder *builder, void **array, size_t *size) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
155
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
156 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
157
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
158 void dbuObjectBuilderDestroy(DBUObjectBuilder *builder) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
159
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
160 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
161
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
162
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
163
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
164 /* ------------------------- Object Builder -----------------------------*/
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
165
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
166 static int add_to_parent(DBUObjectResult *result, DBUObject parent, DBUClass *type, void *obj, CxList *fklist, const CxAllocator *a) {
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
167 DBUBuilderQuery *query = result->userdata1;
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
168 DBUObjectBuilder *builder = result->userdata2;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
169 CxIterator i = cxListIterator(fklist);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
170 cx_foreach(DBUFK*, fk, i) {
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
171 // check if the elm can be added
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
172 DBUField *field = cxMapGet(fk->cls->obj_fields, type->name);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
173 if(!field) {
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
174 continue;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
175 }
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
176
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
177 // find cached object for all foreign keys
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
178 DBUBuilderObjCache *parent_cached = cxMapGet(builder->cache, fk->cache_key);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
179 if(!parent_cached) {
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
180 continue;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
181 }
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
182
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
183 if(field->builder.add) {
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
184 field->builder.add(&field->builder, parent_cached->obj, type, obj, NULL, a);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
185 }
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
186 }
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
187
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
188 return 0;
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
189 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
190
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
191 int dbuObjectExec(DBUObjectBuilder *builder, DBUObjectResult *objresult) {
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
192 if(cxListSize(builder->additionalQueries) > 0 || builder->denseResult) {
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
193 builder->cache = cxHashMapCreateSimple(sizeof(DBUBuilderObjCache));
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
194 if(!builder->cache) {
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
195 return 1;
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
196 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
197 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
198
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
199 int ret = dbuObjectExecuteQuery(builder, builder->mainQuery, builder->resultType, objresult, builder->denseResult);
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
200 if(!ret) {
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
201 CxIterator i = cxListIterator(builder->additionalQueries);
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
202 cx_foreach(DBUBuilderQuery *, q, i) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
203 DBUObjectResult result = {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
204 .userdata1 = q,
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
205 .userdata2 = builder,
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
206 .int1 = CX_STORE_POINTERS,
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
207 .create = result_create_obj,
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
208 .add = add_to_parent
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
209 };
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
210
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
211 ret = dbuObjectExecuteQuery(builder, q->query, q->type, &result, false);
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
212 if(ret) {
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
213 break;
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
214 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
215 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
216 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
217
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
218 if(builder->cache) {
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
219 cxMapDestroy(builder->cache);
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
220 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
221 builder->cache = NULL;
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
222 return ret;
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
223 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
224
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
225 void dbufkelm_free(DBUFK *elm) {
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
226 free(elm->cache_key);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
227 }
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
228
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
229 int dbuObjectExecuteQuery(DBUObjectBuilder *builder, DBUQuery *query, DBUClass *type, DBUObjectResult *objresult, bool dense) {
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
230 DBUClass *cls = type;
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
231
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
232 // TODO: execute additional queries
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
233
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
234 // execute sql
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
235 if(query->exec(query)) {
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
236 query->free(query);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
237 return 1;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
238 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
239
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
240 DBUResult *result = query->getResult(query);
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
241 if(!result) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
242 return 1;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
243 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
244 query->free(query);
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
245
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
246 // map result to class fields
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
247 int numcols = result->numFields(result);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
248 DBUFieldMapping *field_mapping = calloc(numcols, sizeof(DBUFieldMapping));
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
249 if(!field_mapping) {
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
250 result->free(result);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
251 return 1;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
252 }
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
253
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
254 bool is_main = true;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
255 DBUClass *field_class = cls;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
256
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
257 int main_pk_index = -1;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
258 int end_main_fields = numcols;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
259
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
260 for(int i=0;i<numcols;i++) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
261 cxstring fieldname = cx_str(result->fieldName(result, i));
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
262 DBUField *field = NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
263 if(cx_strprefix(fieldname, CX_STR("__"))) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
264 // columns starting with __ are reserved for marking table names
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
265 // __<table>__<fieldname>
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
266 // __<table> (value ignored)
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
267 //
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
268 // after a column is marked as the start of the new table
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
269 // all following columns will be treated as columns of this table
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
270 cxstring tabname = cx_strsubs(fieldname, 2);
5
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
271 cxstring remaining = cx_strstr(tabname, CX_STR("__"));
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
272 DBUClass *fcls = NULL;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
273 if(remaining.length > 2) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
274 tabname.length = remaining.ptr - tabname.ptr;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
275
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
276 }
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
277 fcls = cxMapGet(builder->ctx->classes, tabname);
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
278
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
279 if(fcls) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
280 if(remaining.length > 2) {
5
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
281 field = cxMapGet(fcls->fields, cx_strsubs(remaining, 2));
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
282 }
5
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
283 field_mapping[i].cls = fcls;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
284 field_class = fcls;
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
285 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
286 } else {
5
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
287 field = cxMapGet(field_class->fields, fieldname);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
288 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
289
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
290 if(field) {
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
291 DBUFieldMapping mapping;
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
292 mapping.cls = field_class;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
293 mapping.field = field;
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
294 mapping.type = result->fieldType(result, i);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
295 mapping.is_main = is_main;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
296 field_mapping[i] = mapping;
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
297
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
298 if(field == cls->primary_key) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
299 main_pk_index = i;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
300 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
301
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
302 if(end_main_fields == numcols && field_class != cls) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
303 end_main_fields = i;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
304 }
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
305 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
306 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
307
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
308 if(main_pk_index < 0) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
309 dense = false;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
310 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
311
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
312 const CxAllocator *a = builder->allocator;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
313
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
314 CxList *fklist = cxArrayListCreateSimple(sizeof(DBUFK), 4);
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
315 fklist->collection.simple_destructor = (cx_destructor_func)dbufkelm_free;
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
316
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
317 // get result
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
318 int err = 0;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
319 while(result->hasData(result)) {
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
320
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
321 // create main result obj
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
322 bool addobj = true;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
323 void *obj = NULL;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
324 int skip_fields = 0;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
325 if(dense) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
326 cxstring text = result->getText(result, main_pk_index);
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
327 cxmutstr cache_key = cx_asprintf("%.*s::%.*s",
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
328 (int)cls->name.length, cls->name.ptr,
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
329 (int)text.length, text.ptr);
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
330 if(!cache_key.ptr) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
331 err = 1;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
332 break;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
333 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
334 DBUBuilderObjCache *cached_obj = cxMapGet(builder->cache, cache_key);
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
335 free(cache_key.ptr);
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
336 if(cached_obj && cached_obj->class == cls) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
337 obj = cached_obj->obj;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
338 addobj = false;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
339 skip_fields = end_main_fields;
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
340 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
341 }
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
342
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
343 obj = objresult->create(objresult, cls, a);
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
344 if(!obj) {
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
345 break;
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
346 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
347 memset(obj, 0, sizeof(cls->obj_size));
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
348 if(cls->init) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
349 cls->init(obj, a);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
350 }
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
351
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
352 void *current_obj = obj;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
353 void *child_obj = NULL;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
354 DBUClass *current_cls = cls;
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
355 for(int i=skip_fields;i<numcols;i++) {
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
356 DBUFieldMapping field = field_mapping[i];
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
357 int isnull = result->isNull(result, i);
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
358
5
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
359 if(field.cls && field.cls != current_cls) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
360 // following columns are for this new class
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
361 current_cls = field.cls;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
362 if(field.field) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
363 // check if an object of this class can be added to the main obj
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
364 DBUField *obj_field = cxMapGet(cls->obj_fields, field.cls->name);
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
365 if(obj_field && obj_field->initObjValue) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
366 child_obj = objresult->create(objresult, field.cls, a);
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
367 current_obj = child_obj;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
368 if(!child_obj) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
369 err = 1;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
370 break;
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
371 }
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
372
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
373 if(field.cls->init) {
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
374 field.cls->init(child_obj, a); // TODO: check return
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
375 }
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
376
f0c66b2139ea add support for child objects in the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 4
diff changeset
377 obj_field->initObjValue(obj_field, a, obj, child_obj);
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
378 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
379 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
380 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
381
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
382 DBUField *type_field = field.field;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
383 if(type_field) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
384 if(isnull) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
385 field.field->initDefaultValue(field.field, a, current_obj);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
386 } else {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
387 cxstring text = result->getText(result, i);
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
388 field.field->initValue(field.field, a, current_obj, text.ptr, text.length);
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
389
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
390 // if obj caching is enabled and the current field contains
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
391 // the primary key, add this object to the cache
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
392 if(builder->cache) {
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
393 if(field.field == current_cls->primary_key) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
394 cxmutstr cache_key = cx_asprintf("%.*s::%.*s",
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
395 (int)current_cls->name.length, current_cls->name.ptr,
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
396 (int)text.length, text.ptr);
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
397 if(!cache_key.ptr) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
398 err = 1;
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
399 break;
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
400 }
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
401 DBUBuilderObjCache cache_elm;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
402 cache_elm.class = current_cls;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
403 cache_elm.obj = current_obj;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
404 int r = cxMapPut(builder->cache, cache_key, &cache_elm);
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
405 free(cache_key.ptr);
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
406 if(r) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
407 err = 1;
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
408 break;
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
409 }
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
410 } else if(field.field->foreignKeyClass) {
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
411 cxmutstr fkey = cx_asprintf("%.*s::%.*s",
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
412 (int)field.field->foreignKeyClass->name.length, field.field->foreignKeyClass->name.ptr,
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
413 (int)text.length, text.ptr);
8
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
414 DBUFK fkelm;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
415 fkelm.cache_key = fkey.ptr;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
416 fkelm.cls = field.field->foreignKeyClass;
bd08116b8af4 add support for list members to the object builder
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 7
diff changeset
417 cxListAdd(fklist, &fkelm);
6
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
418 }
d6981b56ab30 add object cache
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 5
diff changeset
419 }
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
420 }
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
421 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
422 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
423
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
424 if(err) {
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
425 break;
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
426 }
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
427
9
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
428 if(addobj) {
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
429 objresult->add(objresult, NULL, cls, obj, fklist, a);
5785a693834c prepare for dense queries
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 8
diff changeset
430 }
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
431
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
432 cxListClear(fklist);
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
433
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
434 // load next row
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
435 result->nextRow(result);
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
436 }
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
437
7
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
438 cxListDestroy(fklist);
c98ff52cd806 prepare for list members
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 6
diff changeset
439
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
440 result->free(result);
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
441
4
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
442
1908c8b1599f add new object builder API
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 3
diff changeset
443 return err;
3
69ea9040d896 add DB abstraction layer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
444 }

mercurial