3 // vformat support routines in base class
7 Copyright (C) 2006-2010, Net Direct Inc. (http://www.netdirect.ca/)
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2 of the License, or
12 (at your option) any later version.
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
18 See the GNU General Public License in the COPYING file at the
19 root directory of this project for more details.
24 #include "vformat.h" // comes from opensync, but not a public header yet
25 #include "tzwrapper.h"
33 namespace Barry
{ namespace Sync
{
35 //////////////////////////////////////////////////////////////////////////////
38 std::string
vTimeConverter::unix2vtime(const time_t *timestamp
)
41 if( !gmtime_r(timestamp
, &split
) ) {
43 oss
<< "gmtime_r() failed on time_t of ";
47 oss
<< "(null pointer)";
48 throw Barry::ConvertError(oss
.str());
51 return tm_to_iso(&split
, true);
54 time_t vTimeConverter::vtime2unix(const char *vtime
)
56 return TzWrapper::iso_mktime(vtime
);
60 // The following implementation is taken from opensync's
61 // opensync_time.c implementation with the following copyright
62 // notices at the top as of July 2010:
64 // * Copyright (C) 2004-2005 Armin Bauer <armin.bauer@opensync.org>
65 // * Copyright (C) 2006-2008 Daniel Gollub <gollub@b1-systems.de>
66 // * Copyright (C) 2007 Chris Frey <cdfrey@netdirect.ca>
68 // License: LGPL 2.1 or later
70 int vTimeConverter::alarmduration2sec(const char *alarm
)
72 int i
, secs
, digits
= 0;
74 int sign
= 1; // when ical stamp doesn't start with '-' => seconds after event
75 int days
= 0, weeks
= 0, hours
= 0, minutes
= 0, seconds
= 0;
76 int len
= strlen(alarm
);
78 for (i
=0; i
< len
; i
++) {
82 sign
= -1; // seconds before event - change the sign
120 if (sscanf((char*)(alarm
+i
),"%d",&digits
) == EOF
)
128 secs
= (weeks
* 7 * 24 * 3600) + (days
* 24 * 3600) + (hours
* 3600) + (minutes
* 60) + seconds
;
130 secs
= secs
* sign
; // change sign if the alarm is in seconds before event (leading '-')
137 //////////////////////////////////////////////////////////////////////////////
140 std::string
vAttr::GetName()
147 const char *name
= b_vformat_attribute_get_name(m_attr
);
153 std::string
vAttr::GetValue(int nth
)
156 const char *value
= 0;
159 if( b_vformat_attribute_is_single_valued(m_attr
) ) {
161 value
= b_vformat_attribute_get_value(m_attr
);
164 value
= b_vformat_attribute_get_nth_value(m_attr
, nth
);
174 std::string
vAttr::GetDecodedValue()
177 GString
*value
= NULL
;
180 if( b_vformat_attribute_is_single_valued(m_attr
) ) {
181 value
= b_vformat_attribute_get_value_decoded(m_attr
);
186 ret
.assign(value
->str
, value
->len
);
191 std::string
vAttr::GetParam(const char *name
, int nth
)
198 b_VFormatParam
*param
= b_vformat_attribute_find_param(m_attr
, name
, 0);
202 const char *value
= b_vformat_attribute_param_get_nth_value(param
, nth
);
209 /// Does an exhaustive search through the attribute, searching for all
210 /// param values that exist for the given name, and returns all values
211 /// in a comma delimited string.
212 std::string
vAttr::GetAllParams(const char *name
)
219 b_VFormatParam
*param
= 0;
221 (param
= b_vformat_attribute_find_param(m_attr
, name
, level
));
224 const char *value
= 0;
226 (value
= b_vformat_attribute_param_get_nth_value(param
, nth
));
239 //////////////////////////////////////////////////////////////////////////////
250 b_vformat_free(m_format
);
255 void vBase::SetFormat(b_VFormat
*format
)
258 b_vformat_free(m_format
);
267 b_vformat_free(m_format
);
272 vAttrPtr
vBase::NewAttr(const char *name
)
274 // Trace trace("vBase::NewAttr");
276 // trace.logf("creating valueless attr: %s", name);
278 vAttrPtr
attr(b_vformat_attribute_new(NULL
, name
));
280 throw Barry::ConvertError("resource error allocating vformat attribute");
284 vAttrPtr
vBase::NewAttr(const char *name
, const char *value
)
286 // Trace trace("vBase::NewAttr");
289 some vCard values are positional (like name), so blank should be allowed...
291 if( strlen(value) == 0 ) {
292 trace.logf("attribute '%s' contains no data, skipping", name);
297 // trace.logf("creating attr: %s, %s", name, value);
299 vAttrPtr
attr(b_vformat_attribute_new(NULL
, name
));
301 throw ConvertError("resource error allocating vformat attribute");
303 b_vformat_attribute_add_value(attr
.Get(), value
);
307 void vBase::AddAttr(vAttrPtr attr
)
309 // Trace trace("vBase::AddAttr");
312 // trace.log("attribute contains no data, skipping");
316 b_vformat_add_attribute(m_format
, attr
.Extract());
319 void vBase::AddValue(vAttrPtr
&attr
, const char *value
)
321 // Trace trace("vBase::AddValue");
323 // trace.log("attribute pointer contains no data, skipping");
327 if( strlen(value) == 0 ) {
328 trace.log("attribute value is empty, skipping");
332 b_vformat_attribute_add_value(attr
.Get(), value
);
335 void vBase::AddEncodedValue(vAttrPtr
&attr
, b_VFormatEncoding encoding
, const char *value
, int len
)
337 // Trace trace("vBase::AddValue");
339 // trace.log("attribute pointer contains no data, skipping");
343 attr
.Get()->encoding
= encoding
;
344 attr
.Get()->encoding_set
= TRUE
;
346 b_vformat_attribute_add_value_decoded(attr
.Get(), value
, len
);
349 void vBase::AddParam(vAttrPtr
&attr
, const char *name
, const char *value
)
351 // Trace trace("vBase::AddParam");
354 // trace.log("attribute pointer contains no data, skipping");
358 if( strlen(value) == 0 ) {
359 trace.log("parameter value is empty, skipping");
364 b_VFormatParam
*pParam
= b_vformat_attribute_param_new(name
);
365 b_vformat_attribute_param_add_value(pParam
, value
);
366 b_vformat_attribute_add_param(attr
.Get(), pParam
);
369 std::string
vBase::GetAttr(const char *attrname
, const char *block
)
371 // Trace trace("vBase::GetAttr");
372 // trace.logf("getting attr: %s", attrname);
375 const char *value
= 0;
377 bool needs_freeing
= false;
379 b_VFormatAttribute
*attr
= b_vformat_find_attribute(m_format
, attrname
, 0, block
);
381 if( b_vformat_attribute_is_single_valued(attr
) ) {
382 value
= b_vformat_attribute_get_value(attr
);
383 needs_freeing
= true;
386 // FIXME, this is hardcoded
387 value
= b_vformat_attribute_get_nth_value(attr
, 0);
395 g_free((char *)value
);
397 // trace.logf("attr value: %s", ret.c_str());
401 std::vector
<std::string
> vBase::GetValueVector(const char *attrname
, const char *block
)
403 // Trace trace("vBase::GetValueVector");
404 // trace.logf("getting value vector for: %s", attrname);
406 std::vector
<std::string
> ret
;
407 const char *value
= 0;
408 bool needs_freeing
= false;
410 b_VFormatAttribute
*attr
= b_vformat_find_attribute(m_format
, attrname
, 0, block
);
412 if( b_vformat_attribute_is_single_valued(attr
) ) {
413 value
= b_vformat_attribute_get_value(attr
);
414 needs_freeing
= true;
416 // nasty, but avoids tweaking vformat.
419 value
= b_vformat_attribute_get_nth_value(attr
, idx
++);
421 ret
.push_back(value
);
428 g_free((char *)value
);
433 vAttr
vBase::GetAttrObj(const char *attrname
, int nth
, const char *block
)
435 // Trace trace("vBase::GetAttrObj");
436 // trace.logf("getting attr: %s", attrname);
438 return vAttr(b_vformat_find_attribute(m_format
, attrname
, nth
, block
));
441 std::vector
<std::string
> vBase::Tokenize(const std::string
& str
, const char delim
)
443 std::vector
<std::string
> tokens
;
444 std::string::size_type delimPos
= 0, tokenPos
= 0, pos
= 0;
446 if( str
.length() < 1 ) {
451 delimPos
= str
.find_first_of(delim
, pos
);
452 tokenPos
= str
.find_first_not_of(delim
, pos
);
454 if( std::string::npos
!= delimPos
) {
455 if( std::string::npos
!= tokenPos
) {
456 if( tokenPos
< delimPos
) {
457 tokens
.push_back(str
.substr(pos
, delimPos
-pos
));
459 tokens
.push_back("");
462 tokens
.push_back("");
466 if( std::string::npos
!= tokenPos
){
467 tokens
.push_back(str
.substr(pos
));
469 tokens
.push_back("");
477 std::string
vBase::ToStringList(const std::vector
<std::string
> &list
, const char delim
)
480 for( unsigned int idx
= 0; idx
< list
.size(); idx
++ ) {
489 }} // namespace Barry::Sync