bump product version to 6.3.0.0.beta1
[LibreOffice.git] / configmgr / source / data.cxx
blob6eb2d5902db3b73f31fc2f2eb2f65522f6a0018d
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <sal/config.h>
22 #include <algorithm>
23 #include <cassert>
25 #include <com/sun/star/uno/Reference.hxx>
26 #include <com/sun/star/uno/RuntimeException.hpp>
27 #include <com/sun/star/uno/XInterface.hpp>
28 #include <rtl/ref.hxx>
29 #include <rtl/string.h>
30 #include <rtl/ustrbuf.hxx>
31 #include <rtl/ustring.h>
32 #include <rtl/ustring.hxx>
33 #include <sal/log.hxx>
34 #include <sal/types.h>
36 #include "additions.hxx"
37 #include "data.hxx"
38 #include "groupnode.hxx"
39 #include "node.hxx"
40 #include "nodemap.hxx"
41 #include "rootnode.hxx"
42 #include "setnode.hxx"
44 namespace configmgr {
46 namespace {
48 bool decode(
49 OUString const & encoded, sal_Int32 begin, sal_Int32 end,
50 OUString * decoded)
52 assert(
53 begin >= 0 && begin <= end && end <= encoded.getLength() &&
54 decoded != nullptr);
55 OUStringBuffer buf;
56 while (begin != end) {
57 sal_Unicode c = encoded[begin++];
58 if (c == '&') {
59 if (encoded.match("amp;", begin)) {
60 buf.append('&');
61 begin += RTL_CONSTASCII_LENGTH("amp;");
62 } else if (encoded.match("quot;", begin)) {
63 buf.append('"');
64 begin += RTL_CONSTASCII_LENGTH("quot;");
65 } else if (encoded.match("apos;", begin)) {
66 buf.append('\'');
67 begin += RTL_CONSTASCII_LENGTH("apos;");
68 } else {
69 return false;
71 assert(begin <= end);
72 } else {
73 buf.append(c);
76 *decoded = buf.makeStringAndClear();
77 return true;
82 OUString Data::createSegment(
83 OUString const & templateName, OUString const & name)
85 if (templateName.isEmpty()) {
86 return name;
88 OUStringBuffer buf(templateName);
89 //TODO: verify template name contains no bad chars?
90 buf.append("['");
91 for (sal_Int32 i = 0; i < name.getLength(); ++i) {
92 sal_Unicode c = name[i];
93 switch (c) {
94 case '&':
95 buf.append("&amp;");
96 break;
97 case '"':
98 buf.append("&quot;");
99 break;
100 case '\'':
101 buf.append("&apos;");
102 break;
103 default:
104 buf.append(c);
105 break;
108 buf.append("']");
109 return buf.makeStringAndClear();
112 sal_Int32 Data::parseSegment(
113 OUString const & path, sal_Int32 index, OUString * name,
114 bool * setElement, OUString * templateName)
116 assert(
117 index >= 0 && index <= path.getLength() && name != nullptr &&
118 setElement != nullptr);
119 sal_Int32 i = index;
120 while (i < path.getLength() && path[i] != '/' && path[i] != '[') {
121 ++i;
123 if (i == path.getLength() || path[i] == '/') {
124 *name = path.copy(index, i - index);
125 *setElement = false;
126 return i;
128 if (templateName != nullptr) {
129 if (i - index == 1 && path[index] == '*') {
130 templateName->clear();
131 } else {
132 *templateName = path.copy(index, i - index);
135 if (++i == path.getLength()) {
136 return -1;
138 sal_Unicode del = path[i++];
139 if (del != '\'' && del != '"') {
140 return -1;
142 sal_Int32 j = path.indexOf(del, i);
143 if (j == -1 || j + 1 == path.getLength() || path[j + 1] != ']' ||
144 !decode(path, i, j, name))
146 return -1;
148 *setElement = true;
149 return j + 2;
152 OUString Data::fullTemplateName(
153 OUString const & component, OUString const & name)
155 if (component.indexOf(':') != -1 || name.indexOf(':') != -1) {
156 throw css::uno::RuntimeException(
157 "bad component/name pair containing colon " + component + "/" +
158 name);
160 OUStringBuffer buf(component);
161 buf.append(':');
162 buf.append(name);
163 return buf.makeStringAndClear();
166 bool Data::equalTemplateNames(
167 OUString const & shortName, OUString const & longName)
169 if (shortName.indexOf(':') == -1) {
170 sal_Int32 i = longName.indexOf(':') + 1;
171 assert(i > 0);
172 return
173 rtl_ustr_compare_WithLength(
174 shortName.getStr(), shortName.getLength(),
175 longName.getStr() + i, longName.getLength() - i) ==
177 } else {
178 return shortName == longName;
182 Data::Data(): root_(new RootNode) {}
184 rtl::Reference< Node > Data::resolvePathRepresentation(
185 OUString const & pathRepresentation,
186 OUString * canonicRepresentation, std::vector<OUString> * path, int * finalizedLayer)
187 const
189 if (pathRepresentation.isEmpty() || pathRepresentation[0] != '/') {
190 throw css::uno::RuntimeException(
191 "bad path " + pathRepresentation);
193 if (path != nullptr) {
194 path->clear();
196 if (pathRepresentation == "/") {
197 if (canonicRepresentation != nullptr) {
198 *canonicRepresentation = pathRepresentation;
200 if (finalizedLayer != nullptr) {
201 *finalizedLayer = NO_LAYER;
203 return root_;
205 OUString seg;
206 bool setElement;
207 OUString templateName;
208 sal_Int32 n = parseSegment(pathRepresentation, 1, &seg, &setElement, nullptr);
209 if (n == -1 || setElement)
211 throw css::uno::RuntimeException(
212 "bad path " + pathRepresentation);
214 NodeMap const & components = getComponents();
215 NodeMap::const_iterator i(components.find(seg));
216 OUStringBuffer canonic;
217 rtl::Reference< Node > parent;
218 int finalized = NO_LAYER;
219 for (rtl::Reference< Node > p(i == components.end() ? nullptr : i->second);;) {
220 if (!p.is()) {
221 return p;
223 if (canonicRepresentation != nullptr) {
224 canonic.append('/');
225 canonic.append(createSegment(templateName, seg));
227 if (path != nullptr) {
228 path->push_back(seg);
230 finalized = std::min(finalized, p->getFinalized());
231 if (n != pathRepresentation.getLength() &&
232 pathRepresentation[n++] != '/')
234 throw css::uno::RuntimeException(
235 "bad path " + pathRepresentation);
237 // for backwards compatibility, ignore a final slash
238 if (n == pathRepresentation.getLength()) {
239 if (canonicRepresentation != nullptr) {
240 *canonicRepresentation = canonic.makeStringAndClear();
242 if (finalizedLayer != nullptr) {
243 *finalizedLayer = finalized;
245 return p;
247 parent = p;
248 templateName.clear();
249 n = parseSegment(
250 pathRepresentation, n, &seg, &setElement, &templateName);
251 if (n == -1) {
252 throw css::uno::RuntimeException(
253 "bad path " + pathRepresentation);
255 // For backwards compatibility, allow set members to be accessed with
256 // simple path segments, like group members:
257 p = p->getMember(seg);
258 if (setElement) {
259 switch (parent->kind()) {
260 case Node::KIND_LOCALIZED_PROPERTY:
261 if (!templateName.isEmpty()) {
262 throw css::uno::RuntimeException(
263 "bad path " + pathRepresentation);
265 break;
266 case Node::KIND_SET:
267 if (!templateName.isEmpty() &&
268 !static_cast< SetNode * >(parent.get())->isValidTemplate(
269 templateName))
271 throw css::uno::RuntimeException(
272 "bad path " + pathRepresentation);
274 break;
275 default:
276 throw css::uno::RuntimeException(
277 "bad path " + pathRepresentation);
279 if (!templateName.isEmpty() && p != nullptr) {
280 assert(!p->getTemplateName().isEmpty());
281 if (!equalTemplateNames(templateName, p->getTemplateName())) {
282 throw css::uno::RuntimeException(
283 "bad path " + pathRepresentation);
290 rtl::Reference< Node > Data::getTemplate(
291 int layer, OUString const & fullName) const
293 return templates.findNode(layer, fullName);
296 NodeMap & Data::getComponents() const {
297 return root_->getMembers();
300 Additions * Data::addExtensionXcuAdditions(
301 OUString const & url, int layer)
303 rtl::Reference item(new ExtensionXcu);
304 ExtensionXcuAdditions::iterator i(
305 extensionXcuAdditions_.emplace(
306 url, rtl::Reference< ExtensionXcu >()).first);
307 if (i->second.is()) {
308 throw css::uno::RuntimeException(
309 "already added extension xcu " + url);
311 i->second = item;
312 item->layer = layer;
313 return &item->additions;
316 rtl::Reference< Data::ExtensionXcu > Data::removeExtensionXcuAdditions(
317 OUString const & url)
319 ExtensionXcuAdditions::iterator i(extensionXcuAdditions_.find(url));
320 if (i == extensionXcuAdditions_.end()) {
321 // This can happen, as migration of pre OOo 3.3 UserInstallation
322 // extensions in dp_registry::backend::configuration::BackendImpl::
323 // PackageImpl::processPackage_ can cause just-in-time creation of
324 // extension xcu files that are never added via addExtensionXcuAdditions
325 // (also, there might be url spelling differences between calls to
326 // addExtensionXcuAdditions and removeExtensionXcuAdditions?):
327 SAL_INFO(
328 "configmgr",
329 "unknown Data::removeExtensionXcuAdditions(" << url << ")");
330 return rtl::Reference< ExtensionXcu >();
332 rtl::Reference< ExtensionXcu > item(i->second);
333 extensionXcuAdditions_.erase(i);
334 return item;
339 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */