Compare commits
20 Commits
Author | SHA1 | Date | |
---|---|---|---|
ce70e5ddd4 | |||
935f1bae2f | |||
0ec1856d1b | |||
ddddcf0af9 | |||
5565087359 | |||
98c300fee2 | |||
daa58518e3 | |||
8e25c3d100 | |||
71107dda1c | |||
cf4c14f9f8 | |||
31f47f60c2 | |||
f59403ad72 | |||
7d642adf2d | |||
3f1efe9e85 | |||
6bbcd9b6a3 | |||
8e58189a8b | |||
9d030132de | |||
68554920e5 | |||
7b71cb2dfe | |||
47c62cce6d |
8
.gitignore
vendored
8
.gitignore
vendored
@ -1,3 +1,7 @@
|
|||||||
|
tests/*
|
||||||
|
!tests/*.*
|
||||||
|
|
||||||
|
|
||||||
|
bin/
|
||||||
|
doc/
|
||||||
*.sw?
|
*.sw?
|
||||||
tests/tlexer
|
|
||||||
tests/tvcard3
|
|
||||||
|
31
Makefile
Normal file
31
Makefile
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
# Make does not offer a recursive wildcard function, so here's one:
|
||||||
|
rwildcard=$(wildcard $1$2) $(foreach d,$(wildcard $1*),$(call rwildcard,$d/,$2))
|
||||||
|
|
||||||
|
SOURCES=$(call rwildcard,src/,*.nim)
|
||||||
|
TEST_SOURCES=$(wildcard tests/*.nim)
|
||||||
|
TESTS=$(patsubst %.nim,bin/%,$(TEST_SOURCES))
|
||||||
|
|
||||||
|
.PHONY: build
|
||||||
|
build: test docs
|
||||||
|
|
||||||
|
doc/vcard/vcard.html: $(SOURCES)
|
||||||
|
nim doc --project --outdir:doc/vcard src/vcard.nim
|
||||||
|
|
||||||
|
.PHONY: doc
|
||||||
|
docs: doc/vcard/vcard.html
|
||||||
|
|
||||||
|
.PHONY: test
|
||||||
|
test:
|
||||||
|
#@for t in $(TESTS); do $$t; done
|
||||||
|
nimble --warning:BareExcept:off test
|
||||||
|
|
||||||
|
.PHONY: install
|
||||||
|
install: test
|
||||||
|
nimble install
|
||||||
|
|
||||||
|
diagrams: doc/vcard3.mmd
|
||||||
|
mmdc -i doc/vcard3.mmd -o doc/vcard3.png
|
||||||
|
|
||||||
|
# Target allowing for running individual tests.
|
||||||
|
bin/tests/%: tests/%.nim $(SOURCES)
|
||||||
|
nim --outdir:bin/tests --hints:off --warning:BareExcept:off c -r $(patsubst bin/%,%.nim,$@)
|
31
README.md
31
README.md
@ -22,30 +22,7 @@ TEL;TYPE=CELL:+1 (555) 123-4567
|
|||||||
END:VCARD
|
END:VCARD
|
||||||
```
|
```
|
||||||
|
|
||||||
```nim
|
https://github.com/jdbernard/nim-vcard/blob/4839ff64a8e6da1ad4803adbd71c0a53cae81c4e/examples/simple.nim#L1-L22
|
||||||
import vcard
|
|
||||||
|
|
||||||
# Reading in an existing vcard
|
|
||||||
let vcards = parseVCard3File("jack.vcf")
|
|
||||||
assert vcards.len == 1
|
|
||||||
let vcAllen = vcards[0]
|
|
||||||
|
|
||||||
assert vcAllen.email.len == 2
|
|
||||||
assert vcAllen.email[0].value == "allen@fosters.test"
|
|
||||||
assert vcAllen.n.first == "Jack"
|
|
||||||
|
|
||||||
|
|
||||||
# Creating a new VCard
|
|
||||||
var vcSusan: VCard3
|
|
||||||
vcSusan.add(
|
|
||||||
newVC3_N(given = "Susan", family = "Foster"),
|
|
||||||
newVC3_Email(value = "susan@fosters.test", emailType = @["PREF", $etInternet),
|
|
||||||
newVC3_Tel(
|
|
||||||
value = "+1 (555) 444-3889",
|
|
||||||
telType = @[$ttHome, $ttCell, $ttVoice, $ttMsg])
|
|
||||||
)
|
|
||||||
writeFile("susan.vcf", $vcSusan)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Future Goals
|
## Future Goals
|
||||||
|
|
||||||
@ -55,9 +32,9 @@ writeFile("susan.vcf", $vcSusan)
|
|||||||
|
|
||||||
*Need to clean up and organize*
|
*Need to clean up and organize*
|
||||||
|
|
||||||
Run `tlexer` tests in gdb:
|
Run `tvcard3` tests in gdb:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
$ cd tests
|
$ cd tests
|
||||||
$ nim --debuginfo --linedir:on c tlexer
|
$ nim --debuginfo --linedir:on c tvcard3
|
||||||
$ gdb --tui tlexer
|
$ gdb --tui tvcard3
|
||||||
|
395
doc/rfc6868.txt
Normal file
395
doc/rfc6868.txt
Normal file
@ -0,0 +1,395 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Internet Engineering Task Force (IETF) C. Daboo
|
||||||
|
Request for Comments: 6868 Apple
|
||||||
|
Updates: 5545, 6321, 6350, 6351 February 2013
|
||||||
|
Category: Standards Track
|
||||||
|
ISSN: 2070-1721
|
||||||
|
|
||||||
|
|
||||||
|
Parameter Value Encoding in iCalendar and vCard
|
||||||
|
|
||||||
|
Abstract
|
||||||
|
|
||||||
|
This specification updates the data formats for iCalendar (RFC 5545)
|
||||||
|
and vCard (RFC 6350) to allow parameter values to include certain
|
||||||
|
characters forbidden by the existing specifications.
|
||||||
|
|
||||||
|
Status of This Memo
|
||||||
|
|
||||||
|
This is an Internet Standards Track document.
|
||||||
|
|
||||||
|
This document is a product of the Internet Engineering Task Force
|
||||||
|
(IETF). It represents the consensus of the IETF community. It has
|
||||||
|
received public review and has been approved for publication by the
|
||||||
|
Internet Engineering Steering Group (IESG). Further information on
|
||||||
|
Internet Standards is available in Section 2 of RFC 5741.
|
||||||
|
|
||||||
|
Information about the current status of this document, any errata,
|
||||||
|
and how to provide feedback on it may be obtained at
|
||||||
|
http://www.rfc-editor.org/info/rfc6868.
|
||||||
|
|
||||||
|
Copyright Notice
|
||||||
|
|
||||||
|
Copyright (c) 2013 IETF Trust and the persons identified as the
|
||||||
|
document authors. All rights reserved.
|
||||||
|
|
||||||
|
This document is subject to BCP 78 and the IETF Trust's Legal
|
||||||
|
Provisions Relating to IETF Documents
|
||||||
|
(http://trustee.ietf.org/license-info) in effect on the date of
|
||||||
|
publication of this document. Please review these documents
|
||||||
|
carefully, as they describe your rights and restrictions with respect
|
||||||
|
to this document. Code Components extracted from this document must
|
||||||
|
include Simplified BSD License text as described in Section 4.e of
|
||||||
|
the Trust Legal Provisions and are provided without warranty as
|
||||||
|
described in the Simplified BSD License.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 1]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
Table of Contents
|
||||||
|
|
||||||
|
1. Introduction ....................................................2
|
||||||
|
2. Conventions Used in This Document ...............................2
|
||||||
|
3. Parameter Value Encoding Scheme .................................3
|
||||||
|
3.1. iCalendar Example ..........................................4
|
||||||
|
3.2. vCard Example ..............................................4
|
||||||
|
4. Security Considerations .........................................4
|
||||||
|
5. Acknowledgments .................................................4
|
||||||
|
6. Normative References ............................................5
|
||||||
|
Appendix A. Choice of Quoting Mechanism ............................6
|
||||||
|
|
||||||
|
1. Introduction
|
||||||
|
|
||||||
|
The iCalendar [RFC5545] specification defines a standard way to
|
||||||
|
describe calendar data. The vCard [RFC6350] specification defines a
|
||||||
|
standard way to describe contact data. Both of these use a similar
|
||||||
|
text-based data format. Each iCalendar and vCard data object can
|
||||||
|
include "properties" that have "parameters" and a "value". The value
|
||||||
|
of a "parameter" is typically a token or URI value, but a "generic"
|
||||||
|
text value is also allowed. However, the syntax rules for both
|
||||||
|
iCalendar and vCard prevent the use of a double-quote character or
|
||||||
|
control characters in such values, though double-quote characters and
|
||||||
|
some subset of control characters are allowed in the actual property
|
||||||
|
values.
|
||||||
|
|
||||||
|
As more and more extensions are being developed for these data
|
||||||
|
formats, there is a need to allow at least double-quotes and line
|
||||||
|
feeds to be included in parameter values. The \-escaping mechanism
|
||||||
|
used for property text values is not defined for use with parameter
|
||||||
|
values and cannot be easily used in a backwards-compatible manner.
|
||||||
|
This specification defines a new character escaping mechanism,
|
||||||
|
compatible with existing parsers and chosen to minimize any impact on
|
||||||
|
existing data.
|
||||||
|
|
||||||
|
2. Conventions Used in This Document
|
||||||
|
|
||||||
|
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT",
|
||||||
|
"SHOULD", "SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and
|
||||||
|
"OPTIONAL" in this document are to be interpreted as described in
|
||||||
|
[RFC2119].
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 2]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
3. Parameter Value Encoding Scheme
|
||||||
|
|
||||||
|
This specification defines the ^ character (U+005E -- Circumflex
|
||||||
|
Accent) as an escape character in parameter values whose value type
|
||||||
|
is defined using the "param-value" syntax element (Section 3.1 of
|
||||||
|
iCalendar [RFC5545] and Section 3.3 of vCard [RFC6350]). The
|
||||||
|
^-escaping mechanism can be used when the value is either unquoted or
|
||||||
|
quoted (i.e., whether or not the value is surrounded by double-
|
||||||
|
quotes).
|
||||||
|
|
||||||
|
When generating iCalendar or vCard parameter values, the following
|
||||||
|
apply:
|
||||||
|
|
||||||
|
o formatted text line breaks are encoded into ^n (U+005E, U+006E)
|
||||||
|
|
||||||
|
o the ^ character (U+005E) is encoded into ^^ (U+005E, U+005E)
|
||||||
|
|
||||||
|
o the " character (U+0022) is encoded into ^' (U+005E, U+0027)
|
||||||
|
|
||||||
|
When parsing iCalendar or vCard parameter values, the following
|
||||||
|
apply:
|
||||||
|
|
||||||
|
o the character sequence ^n (U+005E, U+006E) is decoded into an
|
||||||
|
appropriate formatted line break according to the type of system
|
||||||
|
being used
|
||||||
|
|
||||||
|
o the character sequence ^^ (U+005E, U+005E) is decoded into the ^
|
||||||
|
character (U+005E)
|
||||||
|
|
||||||
|
o the character sequence ^' (U+005E, U+0027) is decoded into the "
|
||||||
|
character (U+0022)
|
||||||
|
|
||||||
|
o if a ^ (U+005E) character is followed by any character other than
|
||||||
|
the ones above, parsers MUST leave both the ^ and the following
|
||||||
|
character in place
|
||||||
|
|
||||||
|
When converting between iCalendar and vCard text-based data formats
|
||||||
|
and alternative data-format representations such as XML (as described
|
||||||
|
in [RFC6321] and [RFC6351], respectively), implementations MUST
|
||||||
|
ensure that parameter value escape sequences are generated correctly
|
||||||
|
in the text-based format and are decoded when the parameter values
|
||||||
|
appear in the alternate data formats.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 3]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
3.1. iCalendar Example
|
||||||
|
|
||||||
|
The following example is an "ATTENDEE" property with a "CN" parameter
|
||||||
|
whose value includes two double-quote characters. The parameter
|
||||||
|
value is not quoted, as there are no characters in the value that
|
||||||
|
would trigger quoting as required by iCalendar.
|
||||||
|
|
||||||
|
ATTENDEE;CN=George Herman ^'Babe^' Ruth:mailto:babe@example.com
|
||||||
|
|
||||||
|
The unescaped parameter value is
|
||||||
|
|
||||||
|
George Herman "Babe" Ruth
|
||||||
|
|
||||||
|
3.2. vCard Example
|
||||||
|
|
||||||
|
The following example is a "GEO" property with an "X-ADDRESS"
|
||||||
|
parameter whose value includes several line feed characters. The
|
||||||
|
parameter value is also quoted, since it contains a comma, which
|
||||||
|
triggers quoting as required by vCard.
|
||||||
|
|
||||||
|
GEO;X-ADDRESS="Pittsburgh Pirates^n115 Federal St^nPitt
|
||||||
|
sburgh, PA 15212":geo:40.446816,-80.00566
|
||||||
|
|
||||||
|
The unescaped parameter value (where each line is terminated by a
|
||||||
|
line break character sequence) is
|
||||||
|
|
||||||
|
Pittsburgh Pirates
|
||||||
|
115 Federal St
|
||||||
|
Pittsburgh, PA 15212
|
||||||
|
|
||||||
|
4. Security Considerations
|
||||||
|
|
||||||
|
There are no additional security issues beyond those of iCalendar
|
||||||
|
[RFC5545] and vCard [RFC6350].
|
||||||
|
|
||||||
|
5. Acknowledgments
|
||||||
|
|
||||||
|
Thanks to Michael Angstadt, Tim Bray, Mike Douglass, Barry Leiba,
|
||||||
|
Simon Perreault, and Pete Resnick for feedback on this specification.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 4]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
6. Normative References
|
||||||
|
|
||||||
|
[RFC2119] Bradner, S., "Key words for use in RFCs to Indicate
|
||||||
|
Requirement Levels", BCP 14, RFC 2119, March 1997.
|
||||||
|
|
||||||
|
[RFC5545] Desruisseaux, B., "Internet Calendaring and Scheduling
|
||||||
|
Core Object Specification (iCalendar)", RFC 5545,
|
||||||
|
September 2009.
|
||||||
|
|
||||||
|
[RFC6321] Daboo, C., Douglass, M., and S. Lees, "xCal: The XML
|
||||||
|
Format for iCalendar", RFC 6321, August 2011.
|
||||||
|
|
||||||
|
[RFC6350] Perreault, S., "vCard Format Specification", RFC 6350,
|
||||||
|
August 2011.
|
||||||
|
|
||||||
|
[RFC6351] Perreault, S., "xCard: vCard XML Representation",
|
||||||
|
RFC 6351, August 2011.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 5]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
Appendix A. Choice of Quoting Mechanism
|
||||||
|
|
||||||
|
Having recognized the need for escaping parameter values, the
|
||||||
|
question is what mechanism to use? One obvious choice would be to
|
||||||
|
adopt the \-escaping used for property values. However, that could
|
||||||
|
not be used as-is, because it escapes a double-quote as the sequence
|
||||||
|
of \ followed by double-quote. Consider what the example in
|
||||||
|
Section 3.1 might look like using \-escaping:
|
||||||
|
|
||||||
|
ATTENDEE;CN="George Herman \"Babe\" Ruth":mailto:babe@example.com
|
||||||
|
|
||||||
|
Existing iCalendar/vCard parsers know nothing about escape sequences
|
||||||
|
in parameters. So they would parse the parameter value as:
|
||||||
|
|
||||||
|
George Herman \
|
||||||
|
|
||||||
|
i.e., the text between the first and second occurrence of a double-
|
||||||
|
quote. However, the text after the second double-quote ought to be
|
||||||
|
either a : or a ; (to delimit the parameter value from the following
|
||||||
|
parameter or property) but is not, so the parser could legitimately
|
||||||
|
throw an error at that point because the data is syntactically
|
||||||
|
invalid. Thus, for backwards-compatibility reasons, a double-quote
|
||||||
|
cannot be escaped using a sequence that itself includes a double-
|
||||||
|
quote, and hence the choice of using a single-quote in this
|
||||||
|
specification.
|
||||||
|
|
||||||
|
Another option would be to use a form of \-escaping modified for use
|
||||||
|
in parameter values only. However, some incorrect, non-interoperable
|
||||||
|
use of \ in parameter values has been observed, and thus it is best
|
||||||
|
to steer clear of that to achieve guaranteed, reliable
|
||||||
|
interoperability. Also, given that double-quote gets changed to
|
||||||
|
single-quote in the escape sequence for a parameter, but not for a
|
||||||
|
value, it is better to not give the impression that the same escape
|
||||||
|
mechanism (and thus code) can be used for both (which could lead to
|
||||||
|
other issues, such as an implementation incorrectly escaping a ; as
|
||||||
|
\; as opposed to quoting the parameter value).
|
||||||
|
|
||||||
|
The choice of ^ as the escape character was made based on the
|
||||||
|
requirement that an ASCII symbol (non-alphanumeric character) be
|
||||||
|
used, and it ought to be one least likely to be found in existing
|
||||||
|
data.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 6]
|
||||||
|
|
||||||
|
RFC 6868 Parameter Encoding February 2013
|
||||||
|
|
||||||
|
|
||||||
|
Author's Address
|
||||||
|
|
||||||
|
Cyrus Daboo
|
||||||
|
Apple Inc.
|
||||||
|
1 Infinite Loop
|
||||||
|
Cupertino, CA 95014
|
||||||
|
USA
|
||||||
|
|
||||||
|
EMail: cyrus@daboo.name
|
||||||
|
URI: http://www.apple.com/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Daboo Standards Track [Page 7]
|
||||||
|
|
10
examples/jack.vcf
Normal file
10
examples/jack.vcf
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
BEGIN:VCARD
|
||||||
|
VERSION:3.0
|
||||||
|
UID: 5db6f100-e2d6-4e8d-951f-d920586bc069
|
||||||
|
N:Foster;Jack;Allen;;
|
||||||
|
FN:Allen Foster
|
||||||
|
REV:20230408T122102Z
|
||||||
|
EMAIL;TYPE=home;TYPE=pref:allen@fosters.test
|
||||||
|
EMAIL;TYPE=work:jack.foster@company.test
|
||||||
|
TEL;TYPE=CELL:+1 (555) 123-4567
|
||||||
|
END:VCARD
|
22
examples/simple.nim
Normal file
22
examples/simple.nim
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import vcard
|
||||||
|
|
||||||
|
# Reading in an existing vcard
|
||||||
|
let vcards = parseVCard3File("jack.vcf")
|
||||||
|
assert vcards.len == 1
|
||||||
|
let vcAllen = vcards[0]
|
||||||
|
|
||||||
|
assert vcAllen.email.len == 2
|
||||||
|
assert vcAllen.email[0].value == "allen@fosters.test"
|
||||||
|
assert vcAllen.n.given[0] == "Jack"
|
||||||
|
|
||||||
|
# Creating a new VCard
|
||||||
|
var vcSusan: VCard3
|
||||||
|
vcSusan.add(@[
|
||||||
|
newVC3_N(given = @["Susan"], family = @["Foster"]),
|
||||||
|
newVC3_Email(value = "susan@fosters.test", emailType = @["PREF",
|
||||||
|
$etInternet]),
|
||||||
|
newVC3_Tel(
|
||||||
|
value = "+1 (555) 444-3889",
|
||||||
|
telType = @[$ttHome, $ttCell, $ttVoice, $ttMsg])
|
||||||
|
])
|
||||||
|
writeFile("susan.vcf", $vcSusan)
|
76
src/vcard.nim
Normal file
76
src/vcard.nim
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
# vCard 3.0 and 4.0 Nim implementation
|
||||||
|
# © 2022 Jonathan Bernard
|
||||||
|
|
||||||
|
## The `vcard` module implements a high-performance vCard parser for both
|
||||||
|
## versions 3.0 (defined by RFCs 2425_ and 2426_) and 4.0 (defined by RFC
|
||||||
|
## 6350_)
|
||||||
|
##
|
||||||
|
## .. _2425: https://tools.ietf.org/html/rfc2425
|
||||||
|
## .. _2426: https://tools.ietf.org/html/rfc2426
|
||||||
|
## .. _6350: https://tools.ietf.org/html/rfc6350
|
||||||
|
import std/[streams, unicode]
|
||||||
|
|
||||||
|
import ./vcard/private/[internals, lexer]
|
||||||
|
import ./vcard/[common, vcard3, vcard4]
|
||||||
|
|
||||||
|
export vcard3, vcard4
|
||||||
|
export common.VC_Param,
|
||||||
|
common.VC_XParam,
|
||||||
|
common.VCard,
|
||||||
|
common.VCardParsingError,
|
||||||
|
common.VCardVersion,
|
||||||
|
common.allPropsOfType,
|
||||||
|
common.getMultipleValues,
|
||||||
|
common.getSingleValue
|
||||||
|
|
||||||
|
proc add*[T](vc: VCard, content: varargs[T]): void =
|
||||||
|
if vc.parsedVersion == VCardV3: add(cast[VCard3](vc), content)
|
||||||
|
else: add(cast[VCard4](vc), content)
|
||||||
|
|
||||||
|
proc readVCard*(p: var VCardParser): VCard =
|
||||||
|
# Read the preamble
|
||||||
|
discard p.readGroup
|
||||||
|
p.expect("begin:vcard" & CRLF)
|
||||||
|
|
||||||
|
# Look for the version tag
|
||||||
|
p.setBookmark
|
||||||
|
discard p.readGroup
|
||||||
|
if p.isNext("version:4.0"):
|
||||||
|
result = VCard4()
|
||||||
|
result.parsedVersion = VCardV4
|
||||||
|
else:
|
||||||
|
result = VCard3()
|
||||||
|
result.parsedVersion = VCardV3
|
||||||
|
p.returnToBookmark
|
||||||
|
|
||||||
|
# VCard3 3.0 allows arbitrarily many empty lines after BEGIN and END
|
||||||
|
if result.parsedVersion == VCardV3:
|
||||||
|
while (p.skip(CRLF, true)): discard
|
||||||
|
for content in vcard3.parseContentLines(p): result.add(content)
|
||||||
|
while (p.skip(CRLF, true)): discard
|
||||||
|
|
||||||
|
else:
|
||||||
|
for content in vcard4.parseContentLines(p): result.add(content)
|
||||||
|
|
||||||
|
if result.parsedVersion == VCardV3:
|
||||||
|
while (p.skip(CRLF, true)): discard
|
||||||
|
|
||||||
|
proc initVCardParser*(input: Stream, filename = "input"): VCardParser =
|
||||||
|
result.filename = filename
|
||||||
|
lexer.open(result, input)
|
||||||
|
|
||||||
|
proc initVCardParser*(content: string, filename = "input"): VCardParser =
|
||||||
|
initVCardParser(newStringStream(content), filename)
|
||||||
|
|
||||||
|
proc initVCardParserFromFile*(filepath: string): VCardParser =
|
||||||
|
initVCardParser(newFileStream(filepath, fmRead), filepath)
|
||||||
|
|
||||||
|
proc parseVCards*(input: Stream, filename = "input"): seq[VCard] =
|
||||||
|
var p = initVCardParser(input, filename)
|
||||||
|
while p.peek != '\0': result.add(p.readVCard)
|
||||||
|
|
||||||
|
proc parseVCards*(content: string, filename = "input"): seq[VCard] =
|
||||||
|
parseVCards(newStringStream(content), filename)
|
||||||
|
|
||||||
|
proc parseVCardsFromFile*(filepath: string): seq[VCard] =
|
||||||
|
parseVCards(newFileStream(filepath, fmRead), filepath)
|
91
src/vcard/common.nim
Normal file
91
src/vcard/common.nim
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
# Common Functionality
|
||||||
|
# © 2022-2023 Jonathan Bernard
|
||||||
|
|
||||||
|
## This module contains type definitions and func/proc definitions that are
|
||||||
|
## used in common between both the 3.0 and 4.0 parser implementations.
|
||||||
|
import std/[options, sequtils]
|
||||||
|
import zero_functional
|
||||||
|
import ./private/lexer
|
||||||
|
|
||||||
|
type
|
||||||
|
VC_Param* = tuple[name: string, values: seq[string]]
|
||||||
|
## Representation of vCard parameter and its values.
|
||||||
|
|
||||||
|
VCardVersion* = enum
|
||||||
|
## enum used to differentiate VCard3 and VCard4 versions.
|
||||||
|
VCardV3 = "3.0", VCardV4 = "4.0"
|
||||||
|
|
||||||
|
VCardParser* = object of VCardLexer
|
||||||
|
## Common vCard parser object
|
||||||
|
filename*: string
|
||||||
|
|
||||||
|
VCardParsingError* = object of ValueError
|
||||||
|
## Error raised when invalid input is detected while parsing a vCard
|
||||||
|
|
||||||
|
VC_XParam* = tuple[name, value: string]
|
||||||
|
## Representation of vCard extended parameters (starting with "X-").
|
||||||
|
## Because the meaning of these parameters is implementation-specific, no
|
||||||
|
## parsing of the parameter value is performed, it is returned verbatim.
|
||||||
|
|
||||||
|
VCard* = ref object of RootObj
|
||||||
|
## Abstract base class for all vCards. `parsedVersion` can be used to
|
||||||
|
## interrogate any concrete instance of this class. `asVCard3` and
|
||||||
|
## `asVCard4` exist as convenience functions to cast an instance to one of
|
||||||
|
## the subclasses depending on the value of `parsedVersion`.
|
||||||
|
parsedVersion*: VCardVersion
|
||||||
|
|
||||||
|
proc getMultipleValues*(
|
||||||
|
params: openarray[VC_Param],
|
||||||
|
name: string
|
||||||
|
): seq[string] =
|
||||||
|
|
||||||
|
## Get all of the values for a given parameter in a single list. There are
|
||||||
|
## two patterns for multi-valued parameters defined in the vCard 3.0 RFCs:
|
||||||
|
##
|
||||||
|
## - TYPE=work,cell,voice
|
||||||
|
## - TYPE=work;TYPE=cell;TYPE=voice
|
||||||
|
##
|
||||||
|
## Parameter values can be specific using both patterns. This method joins
|
||||||
|
## all defined values regardless of the pattern used to define them.
|
||||||
|
|
||||||
|
let ps = params.toSeq
|
||||||
|
ps -->
|
||||||
|
filter(it.name == name).
|
||||||
|
map(it.values).
|
||||||
|
flatten()
|
||||||
|
|
||||||
|
proc getSingleValue*(
|
||||||
|
params: openarray[VC_Param],
|
||||||
|
name: string
|
||||||
|
): Option[string] =
|
||||||
|
## Get the first single value defined for a parameter.
|
||||||
|
##
|
||||||
|
## Many parameters only support a single value, depending on the content type.
|
||||||
|
## In order to support multi-valued parameters our implementation stores all
|
||||||
|
## parameters as seq[string]. This function is a convenience around that.
|
||||||
|
|
||||||
|
let ps = params.toSeq
|
||||||
|
let foundParam = ps --> find(it.name == name)
|
||||||
|
|
||||||
|
if foundParam.isSome and foundParam.get.values.len > 0:
|
||||||
|
return some(foundParam.get.values[0])
|
||||||
|
else:
|
||||||
|
return none[string]()
|
||||||
|
|
||||||
|
func allPropsOfType*[T, VC: VCard](vc: VC): seq[T] =
|
||||||
|
## Get all instances of the requested property type present on the given
|
||||||
|
## vCard.
|
||||||
|
##
|
||||||
|
## This can be useful when there is some logic that hides multiple instances
|
||||||
|
## of a property, or returns a limited subset. For example, on 3.0 versions
|
||||||
|
## of vCards, this library assumes that there will only be one instance of
|
||||||
|
## the NAME property. The 3.0 spec implies that the NAME property should only
|
||||||
|
## be present at most once, but does not explicitly state this. It is
|
||||||
|
## possible for a 3.0 vCard to contain multiple NAME properties. using
|
||||||
|
## `vc3.name` will only return the first. This function allows a caller to
|
||||||
|
## retrieve all instances for any given property type. For example:
|
||||||
|
##
|
||||||
|
## .. code-block:: nim
|
||||||
|
## let vc3 = parseVCards(...)
|
||||||
|
## let allNames = allPropsOfType[VC3_Name](vc3)
|
||||||
|
vc.content.filterIt(it of typeof(T)).mapIt(cast[T](it))
|
283
src/vcard/private/internals.nim
Normal file
283
src/vcard/private/internals.nim
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
# Common Functionality
|
||||||
|
# © 2022-2023 Jonathan Bernard
|
||||||
|
|
||||||
|
## This module contains type definitions, constant values, and func/proc
|
||||||
|
## definitions that are used in common between both the 3.0 and 4.0 parser
|
||||||
|
## implementations.
|
||||||
|
##
|
||||||
|
## This module is not intended to be exposed to library consumers. It is
|
||||||
|
## intended to be imported by the vcard3 and vcard4 implementations.
|
||||||
|
|
||||||
|
import std/[macros, options, strutils, times, unicode]
|
||||||
|
import zero_functional
|
||||||
|
from std/sequtils import toSeq
|
||||||
|
|
||||||
|
import ./lexer
|
||||||
|
import ../common
|
||||||
|
|
||||||
|
# Internal Types (used by `vcard{3,4}`)
|
||||||
|
# =====================================
|
||||||
|
type
|
||||||
|
VC_PropCardinality* = enum
|
||||||
|
## enum used to define the possible cardinalities of vCard properties.
|
||||||
|
vpcAtMostOne,
|
||||||
|
vpcExactlyOne,
|
||||||
|
vpcAtLeastOne
|
||||||
|
vpcAny
|
||||||
|
|
||||||
|
# Internal constants (used by `vcard{3,4}`)
|
||||||
|
# =========================================
|
||||||
|
const CRLF* = "\r\n"
|
||||||
|
const WSP* = {' ', '\t'}
|
||||||
|
const DIGIT* = { '0'..'9' }
|
||||||
|
const ALPHA_NUM* = { 'a'..'z', 'A'..'Z', '0'..'9' }
|
||||||
|
const NON_ASCII* = { '\x80'..'\xFF' }
|
||||||
|
const QSAFE_CHARS* = WSP + { '\x21', '\x23'..'\x7E' } + NON_ASCII
|
||||||
|
const SAFE_CHARS* = WSP + { '\x21', '\x23'..'\x2B', '\x2D'..'\x39', '\x3C'..'\x7E' } + NON_ASCII
|
||||||
|
const VALUE_CHAR* = WSP + { '\x21'..'\x7E' } + NON_ASCII
|
||||||
|
|
||||||
|
const DATE_FMTS = [ "yyyy-MM-dd", "yyyyMMdd" ]
|
||||||
|
const DATE_TIME_FMTS = [
|
||||||
|
"yyyyMMdd'T'HHmmss",
|
||||||
|
"yyyyMMdd'T'HHmmssz",
|
||||||
|
"yyyyMMdd'T'HHmmsszzz",
|
||||||
|
"yyyyMMdd'T'HHmmss'.'fffzzz",
|
||||||
|
"yyyy-MM-dd'T'HH:mm:ss",
|
||||||
|
"yyyy-MM-dd'T'HH:mm:ssz",
|
||||||
|
"yyyy-MM-dd'T'HH:mm:sszzz",
|
||||||
|
"yyyy-MM-dd'T'HH:mm:ss'.'fffzzz",
|
||||||
|
]
|
||||||
|
|
||||||
|
const ALL_DATE_AND_OR_TIME_FMTS = DATE_TIME_FMTS.toSeq & DATE_FMTS.toSeq
|
||||||
|
|
||||||
|
# Internal Utility/Implementation Functions
|
||||||
|
# =========================================
|
||||||
|
|
||||||
|
proc parseDateTimeStr(
|
||||||
|
dateStr: string,
|
||||||
|
dateFmts: openarray[string]
|
||||||
|
): DateTime {.inline, raises:[ValueError].} =
|
||||||
|
|
||||||
|
for fmt in dateFmts:
|
||||||
|
try: result = parse(dateStr, fmt)
|
||||||
|
except ValueError: discard
|
||||||
|
|
||||||
|
if not result.isInitialized:
|
||||||
|
raise newException(ValueError, "cannot parse date: " & dateStr )
|
||||||
|
|
||||||
|
proc parseDate*(dateStr: string): DateTime =
|
||||||
|
parseDateTimeStr(dateStr, DATE_FMTS)
|
||||||
|
|
||||||
|
proc parseDateTime*(dateStr: string): DateTime =
|
||||||
|
parseDateTimeStr(dateStr, DATE_TIME_FMTS)
|
||||||
|
|
||||||
|
proc parseDateOrDateTime*(dateStr: string): DateTime =
|
||||||
|
parseDateTimeStr(dateStr, ALL_DATE_AND_OR_TIME_FMTS)
|
||||||
|
|
||||||
|
template indexOfIt*(s, pred: untyped): int =
|
||||||
|
var result = -1
|
||||||
|
for idx, it {.inject.} in pairs(s):
|
||||||
|
if pred:
|
||||||
|
result = idx
|
||||||
|
break
|
||||||
|
result
|
||||||
|
|
||||||
|
template findAll*[T, VCT](c: openarray[VCT]): seq[T] =
|
||||||
|
c.filterIt(it of typeof(T)).mapIt(cast[T](it))
|
||||||
|
|
||||||
|
template findFirst*[T, VCT](c: openarray[VCT]): Option[T] =
|
||||||
|
let found = c.filterIt(it of typeof(T)).mapIt(cast[T](it))
|
||||||
|
if found.len > 0: some(found[0])
|
||||||
|
else: none[T]()
|
||||||
|
|
||||||
|
macro assignFields*(assign: untyped, fields: varargs[untyped]): untyped =
|
||||||
|
result = assign
|
||||||
|
|
||||||
|
for f in fields:
|
||||||
|
let exp = newNimNode(nnkExprColonExpr)
|
||||||
|
exp.add(f, f)
|
||||||
|
result.add(exp)
|
||||||
|
|
||||||
|
# Internal Parsing Functionality
|
||||||
|
# ==============================
|
||||||
|
|
||||||
|
proc error*(p: VCardParser, msg: string) =
|
||||||
|
raise newException(VCardParsingError, "$1($2, $3) Error: $4" %
|
||||||
|
[ p.filename, $p.lineNumber, $p.getColNumber(p.pos), msg ])
|
||||||
|
|
||||||
|
proc expect*(p: var VCardParser, expected: string, caseSensitive = false) =
|
||||||
|
## Read `expected.len` from the parser's input stream and determine if it
|
||||||
|
## matches the expected value. If the parser is unable to supply
|
||||||
|
## `expected.len` bytes or if the value read does not match the expected
|
||||||
|
## value, raise a VCardParsingError via the `error` utility function
|
||||||
|
## reporting to the user where the input failed to match the expected value.
|
||||||
|
## If the expectation is met, the parser read position is advanced past the
|
||||||
|
## expected value, on the next byte after the expected value that was read.
|
||||||
|
try:
|
||||||
|
p.setBookmark
|
||||||
|
|
||||||
|
if caseSensitive:
|
||||||
|
for ch in expected:
|
||||||
|
if p.read != ch: raise newException(ValueError, "")
|
||||||
|
else:
|
||||||
|
for rune in expected.runes:
|
||||||
|
if p.readRune.toLower != rune.toLower:
|
||||||
|
raise newException(ValueError, "")
|
||||||
|
|
||||||
|
except ValueError:
|
||||||
|
p.error("expected '$1' but found '$2':\n\t$3\n\t$4" %
|
||||||
|
[expected, p.readSinceBookmark, p.lineVal,
|
||||||
|
" ".repeat(p.getColNumber(p.pos) - 1) & "^\n"])
|
||||||
|
|
||||||
|
finally: p.unsetBookmark
|
||||||
|
|
||||||
|
proc isNext*(p: var VCardParser, expected: string, caseSensitive = false): bool =
|
||||||
|
## Read `expected.len` from the parser's input stream and determine if it
|
||||||
|
## matches the expected value. Assuming the parser did not fail to read
|
||||||
|
## `expected.len` bytes, this will reset the parser read state to its initial
|
||||||
|
## position (prior to calling `isNext`). This is similar to `expect` but
|
||||||
|
## exhibits more "`peek`-like" behavior (doesn't disturb the read position,
|
||||||
|
## and doesn't fail).
|
||||||
|
result = true
|
||||||
|
p.setBookmark
|
||||||
|
|
||||||
|
if caseSensitive:
|
||||||
|
for ch in expected:
|
||||||
|
if p.read != ch:
|
||||||
|
result = false
|
||||||
|
break
|
||||||
|
|
||||||
|
else:
|
||||||
|
for rune in expected.runes:
|
||||||
|
if p.readRune.toLower != rune.toLower:
|
||||||
|
result = false
|
||||||
|
break
|
||||||
|
|
||||||
|
p.returnToBookmark
|
||||||
|
|
||||||
|
proc readGroup*(p: var VCardParser): Option[string] =
|
||||||
|
## All vCard content items can be optionally prefixed with a group name. This
|
||||||
|
## scans the input to see if there is a group defined at the current read
|
||||||
|
## location. If there is a valid group, the group name is returned and the
|
||||||
|
## read position is advanced past the '.' to the start of the content type
|
||||||
|
## name. If there is not a valid group the read position is left unchanged.
|
||||||
|
|
||||||
|
p.setBookmark
|
||||||
|
var ch = p.read
|
||||||
|
while ALPHA_NUM.contains(ch): ch = p.read
|
||||||
|
|
||||||
|
if (ch == '.'):
|
||||||
|
result = some(readSinceBookmark(p)[0..^2])
|
||||||
|
p.unsetBookmark
|
||||||
|
else:
|
||||||
|
result = none[string]()
|
||||||
|
p.returnToBookmark
|
||||||
|
|
||||||
|
proc readName*(p: var VCardParser): string =
|
||||||
|
## Read a name from the current read position or error. As both content types
|
||||||
|
## and paramaters use the same definition for valid names, this method is
|
||||||
|
## used to read in both.
|
||||||
|
p.setBookmark
|
||||||
|
let validChars = ALPHA_NUM + {'-'}
|
||||||
|
while validChars.contains(p.peek): discard p.read
|
||||||
|
result = p.readSinceBookmark.toUpper()
|
||||||
|
if result.len == 0:
|
||||||
|
p.error("expected to read a name but found '$1'" % [$p.peek])
|
||||||
|
p.unsetBookmark
|
||||||
|
|
||||||
|
proc readValue*(p: var VCardParser): string =
|
||||||
|
## Read a content value at the current read position.
|
||||||
|
p.setBookmark
|
||||||
|
while VALUE_CHAR.contains(p.peek): discard p.read
|
||||||
|
result = p.readSinceBookmark
|
||||||
|
p.unsetBookmark
|
||||||
|
|
||||||
|
proc skip*(p: var VCardParser, count: int): bool =
|
||||||
|
## Ignore the next `count` bytes of data from the parser's underlying input
|
||||||
|
## stream.
|
||||||
|
for _ in 0..<count: discard p.read
|
||||||
|
|
||||||
|
proc skip*(p: var VCardParser, expected: string, caseSensitive = false): bool =
|
||||||
|
|
||||||
|
## Ignore the next `expected.len` bytes of data from the parser's underlying
|
||||||
|
## input stream, but only if the value read matches the value provided in
|
||||||
|
## `expected`. In other words: read `expected.len` bytes. If they match the
|
||||||
|
## expectation, leave the parser read position in the new state. If they do
|
||||||
|
## not match, reset the parser read position to the state prior to invoking
|
||||||
|
## `skip`.
|
||||||
|
|
||||||
|
p.setBookmark
|
||||||
|
if caseSensitive:
|
||||||
|
for ch in expected:
|
||||||
|
if p.read != ch:
|
||||||
|
p.returnToBookmark
|
||||||
|
return false
|
||||||
|
|
||||||
|
else:
|
||||||
|
for rune in expected.runes:
|
||||||
|
if p.readRune.toLower != rune.toLower:
|
||||||
|
p.returnToBookmark
|
||||||
|
return false
|
||||||
|
|
||||||
|
p.unsetBookmark
|
||||||
|
return true
|
||||||
|
|
||||||
|
proc existsWithValue*(
|
||||||
|
params: openarray[VC_Param],
|
||||||
|
name, value: string,
|
||||||
|
caseSensitive = false
|
||||||
|
): bool =
|
||||||
|
|
||||||
|
## Determine if the given parameter exists and has the expected value. By
|
||||||
|
## default, value checks are not case-sensitive, as most VCard3 values are not
|
||||||
|
## defined as being case-sensitive.
|
||||||
|
|
||||||
|
let ps = params.toSeq
|
||||||
|
|
||||||
|
if caseSensitive:
|
||||||
|
ps --> exists(
|
||||||
|
it.name == name and
|
||||||
|
it.values.len == 1 and
|
||||||
|
it.values[0] == value)
|
||||||
|
else:
|
||||||
|
ps --> exists(
|
||||||
|
it.name == name and
|
||||||
|
it.values.len == 1 and
|
||||||
|
it.values[0].toLower == value.toLower)
|
||||||
|
|
||||||
|
proc validateNoParameters*(
|
||||||
|
p: VCardParser,
|
||||||
|
params: openarray[VC_Param],
|
||||||
|
name: string
|
||||||
|
) =
|
||||||
|
|
||||||
|
## Raise a VCardParsingError if there are any parameters.
|
||||||
|
if params.len > 0:
|
||||||
|
p.error("no parameters allowed on the $1 content type" % [name])
|
||||||
|
|
||||||
|
proc validateRequiredParameters*(
|
||||||
|
p: VCardParser,
|
||||||
|
params: openarray[VC_Param],
|
||||||
|
expectations: openarray[tuple[name: string, value: string]]
|
||||||
|
) =
|
||||||
|
|
||||||
|
## Some content types have specific allowed parameters. For example, the
|
||||||
|
## SOURCE content type requires that the VALUE parameter be set to "uri" if
|
||||||
|
## it is present. This will raise a VCardParsingError if given parameters are
|
||||||
|
## present with different values that expected.
|
||||||
|
|
||||||
|
for (n, v) in expectations:
|
||||||
|
let pv = params.getSingleValue(n)
|
||||||
|
if pv.isSome and pv.get != v:
|
||||||
|
p.error("parameter '$1' must have the value '$2'" % [n, v])
|
||||||
|
|
||||||
|
# Internal Serialization Utilities
|
||||||
|
# ================================
|
||||||
|
|
||||||
|
func foldContentLine*(s: string): string =
|
||||||
|
result = ""
|
||||||
|
var rem = s
|
||||||
|
while rem.len > 75: # TODO: unicode multi-byte safe?
|
||||||
|
result &= rem[0..<75] & "\r\n "
|
||||||
|
rem = rem[75..^1]
|
||||||
|
result &= rem
|
@ -1,25 +1,45 @@
|
|||||||
|
# vCard-specific Lexer
|
||||||
|
# © 2022-2023 Jonathan Bernard
|
||||||
|
|
||||||
|
## This module defines a lexer with functionality useful for parsing vCard
|
||||||
|
## content. Specifically:
|
||||||
|
## - it understands the vCard line-folding logic and transparently joins folded
|
||||||
|
## lines as it read input off its input stream.
|
||||||
|
## - it supports multiple nested bookmarks to make look-ahead decisions more
|
||||||
|
## convenient
|
||||||
|
## - it supports reading from the underlying stream byte-wise or as unicode
|
||||||
|
## runes.
|
||||||
|
##
|
||||||
|
## This parser uses a ring buffer underneath, only growing the size of its
|
||||||
|
## buffer when it is completely full.
|
||||||
|
|
||||||
import std/[streams, unicode]
|
import std/[streams, unicode]
|
||||||
|
|
||||||
type VCardLexer* = object of RootObj
|
type VCardLexer* = object of RootObj
|
||||||
input: Stream
|
input: Stream
|
||||||
|
|
||||||
buffer*: string # buffer of bytes read
|
buffer*: string ## buffer of bytes read
|
||||||
bufStart: int # starting boundary for the buffer
|
bufStart: int ## starting boundary for the buffer
|
||||||
bufEnd: int # ending boundary for the buffer
|
bufEnd: int ## ending boundary for the buffer
|
||||||
pos*: int # current read position
|
pos*: int ## current read position
|
||||||
bookmark*: int # bookmark to support rewind functionality
|
bookmark*: seq[int] ## bookmark to support rewind functionality
|
||||||
bookmarkVal*: string # value that has been read since the bookmark was set
|
bookmarkVal*: seq[string] ## value read since the bookmark was set
|
||||||
lineNumber*: int # how many newlines have we seen so far
|
lineNumber*: int ## how many newlines have we seen so far
|
||||||
lineStart: int # index into the buffer for the start of the current line
|
lineStart: int ## buffer index buffer for the start of the current line
|
||||||
|
lineVal*: string ## value read since the start of the current line
|
||||||
|
|
||||||
proc skipUtf8Bom(vcl: var VCardLexer) =
|
proc skipUtf8Bom(vcl: var VCardLexer) =
|
||||||
if (vcl.buffer[0] == '\xEF') and (vcl.buffer[1] == '\xBB') and (vcl.buffer[2] == '\xBF'):
|
if (vcl.buffer[0] == '\xEF') and (vcl.buffer[1] == '\xBB') and (vcl.buffer[2] == '\xBF'):
|
||||||
inc(vcl.pos, 3)
|
inc(vcl.pos, 3)
|
||||||
|
|
||||||
template wrappedIdx(idx: untyped): int = idx mod vcl.buffer.len
|
template wrappedIdx(idx: untyped): int = idx mod vcl.buffer.len
|
||||||
|
## Map an index into the buffer bounds (mod)
|
||||||
|
|
||||||
proc newStartIdx(vcl: VCardLexer): int =
|
proc newStartIdx(vcl: VCardLexer): int =
|
||||||
if vcl.bookmark > 0: vcl.bookmark else: vcl.pos
|
## Get the latest safe index to use as a new start index. The implication is
|
||||||
|
## that anything prior to this index has been read and processed and can be
|
||||||
|
## safely overwritten in the buffer.
|
||||||
|
if vcl.bookmark.len > 0: vcl.bookmark[0] else: vcl.pos
|
||||||
|
|
||||||
func isFull(vcl: VCardLexer): bool {.inline.} =
|
func isFull(vcl: VCardLexer): bool {.inline.} =
|
||||||
return wrappedIdx(vcl.bufEnd + 1) == vcl.newStartIdx
|
return wrappedIdx(vcl.bufEnd + 1) == vcl.newStartIdx
|
||||||
@ -28,6 +48,8 @@ func atEnd(vcl: VCardLexer): bool {.inline.} =
|
|||||||
vcl.pos == vcl.bufEnd
|
vcl.pos == vcl.bufEnd
|
||||||
|
|
||||||
proc doubleBuffer(vcl: var VCardLexer) =
|
proc doubleBuffer(vcl: var VCardLexer) =
|
||||||
|
## Double the capacity of the buffer, copying the contents of the current
|
||||||
|
## buffer into the beginning of the newly expanded buffer.
|
||||||
let oldBuf = vcl.buffer
|
let oldBuf = vcl.buffer
|
||||||
vcl.buffer = newString(oldBuf.len * 2)
|
vcl.buffer = newString(oldBuf.len * 2)
|
||||||
|
|
||||||
@ -39,13 +61,28 @@ proc doubleBuffer(vcl: var VCardLexer) =
|
|||||||
inc(newIdx)
|
inc(newIdx)
|
||||||
oldIdx = (newIdx + vcl.bufStart) mod oldBuf.len
|
oldIdx = (newIdx + vcl.bufStart) mod oldBuf.len
|
||||||
|
|
||||||
|
# We know that for all existing indices, their location in the new buffer can
|
||||||
|
# be calculated as a function of the distance we moved the start of the
|
||||||
|
# buffer: `idx = idx + (newBufStart - oldBufStart)`. Since we know the new
|
||||||
|
# bufStart will be 0, we know that we can calculate all of the new indices as
|
||||||
|
# `idx -= oldBufStart` Currently vcl.bufStart is still the old bufStart.
|
||||||
vcl.pos -= vcl.bufStart
|
vcl.pos -= vcl.bufStart
|
||||||
vcl.lineStart -= vcl.bufStart
|
vcl.lineStart -= vcl.bufStart
|
||||||
if vcl.bookmark >= 0: vcl.bookmark -= vcl.bufStart
|
if vcl.bookmark.len > 0: vcl.bookmark[0] -= vcl.bufStart
|
||||||
|
|
||||||
|
# Now that we've updated all of the existing indices, we can reset the
|
||||||
|
# buffer start and end indices to their new values.
|
||||||
vcl.bufStart = 0
|
vcl.bufStart = 0
|
||||||
vcl.bufEnd = newIdx
|
vcl.bufEnd = newIdx
|
||||||
|
|
||||||
proc fillBuffer(vcl: var VCardLexer) =
|
proc fillBuffer(vcl: var VCardLexer) =
|
||||||
|
## Read data into the buffer from the underlying stream until the buffer is
|
||||||
|
## full. If the buffer is already full, double the buffer beforehand.
|
||||||
|
|
||||||
|
# Note that we do not *completely* fill the buffer. We always leave one index
|
||||||
|
# of the array empty. This allows us to differentiate between an empty buffer
|
||||||
|
# (`bufStart == bufEnd`) and a completly full buffer (`bufStart ==
|
||||||
|
# wrappedIdx(bufEnd + 1)`).
|
||||||
|
|
||||||
var charsRead: int
|
var charsRead: int
|
||||||
|
|
||||||
@ -55,7 +92,10 @@ proc fillBuffer(vcl: var VCardLexer) =
|
|||||||
# discard used portions of the buffer
|
# discard used portions of the buffer
|
||||||
vcl.bufStart = vcl.newStartIdx
|
vcl.bufStart = vcl.newStartIdx
|
||||||
|
|
||||||
|
# We have three conditions that the ring buffer may be in:
|
||||||
if vcl.bufEnd < vcl.bufStart:
|
if vcl.bufEnd < vcl.bufStart:
|
||||||
|
# The unused portion of the buffer is all in the middle and we can just
|
||||||
|
# read date into the space from bufEnd (e) to bufStart (s).
|
||||||
# e s
|
# e s
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
charsRead = vcl.input.readDataStr(vcl.buffer,
|
charsRead = vcl.input.readDataStr(vcl.buffer,
|
||||||
@ -63,6 +103,8 @@ proc fillBuffer(vcl: var VCardLexer) =
|
|||||||
vcl.bufEnd += charsRead
|
vcl.bufEnd += charsRead
|
||||||
|
|
||||||
elif vcl.bufStart == 0:
|
elif vcl.bufStart == 0:
|
||||||
|
# The unused portion is entirely at the end of the buffer. We can read data
|
||||||
|
# from the bufEnd (e) to the end of our buffer capacity.
|
||||||
# s e
|
# s e
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
charsRead = vcl.input.readDataStr(vcl.buffer,
|
charsRead = vcl.input.readDataStr(vcl.buffer,
|
||||||
@ -70,21 +112,29 @@ proc fillBuffer(vcl: var VCardLexer) =
|
|||||||
vcl.bufEnd = wrappedIdx(vcl.bufEnd + charsRead)
|
vcl.bufEnd = wrappedIdx(vcl.bufEnd + charsRead)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
# The used portion of the buffer is in the middle, and the unused portion
|
||||||
|
# is on either side ot that. We need to read from bufEnd (e) to the end of
|
||||||
|
# our underlying buffer, and then from the start of our underlying buffer
|
||||||
|
# to bufStart (s)
|
||||||
# s e
|
# s e
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
charsRead = vcl.input.readDataStr(vcl.buffer, vcl.bufEnd..<vcl.buffer.len)
|
charsRead = vcl.input.readDataStr(vcl.buffer, vcl.bufEnd..<vcl.buffer.len)
|
||||||
if charsRead == vcl.buffer.len - vcl.bufEnd:
|
if charsRead == vcl.buffer.len - vcl.bufEnd:
|
||||||
|
# Only read into the front part of the buffer if we were able to
|
||||||
|
# completely fill the back part.
|
||||||
vcl.bufEnd = vcl.input.readDataStr(vcl.buffer, 0 ..< (vcl.bufStart - 1))
|
vcl.bufEnd = vcl.input.readDataStr(vcl.buffer, 0 ..< (vcl.bufStart - 1))
|
||||||
|
|
||||||
|
|
||||||
proc close*(vcl: var VCardLexer) = vcl.input.close
|
proc close*(vcl: var VCardLexer) = vcl.input.close
|
||||||
|
## Close this VCardLexer and its underlying stream.
|
||||||
|
|
||||||
proc open*(vcl: var VCardLexer, input: Stream, bufLen = 16384) =
|
proc open*(vcl: var VCardLexer, input: Stream, bufLen = 16384) =
|
||||||
|
## Open the given stream and initialize the given VCardLexer to read from it.
|
||||||
assert(bufLen > 0)
|
assert(bufLen > 0)
|
||||||
assert(input != nil)
|
assert(input != nil)
|
||||||
vcl.input = input
|
vcl.input = input
|
||||||
vcl.pos = 0
|
vcl.pos = 0
|
||||||
vcl.bookmark = -1
|
vcl.bookmark = @[]
|
||||||
vcl.buffer = newString(bufLen)
|
vcl.buffer = newString(bufLen)
|
||||||
vcl.bufStart = 0
|
vcl.bufStart = 0
|
||||||
vcl.bufEnd = 0
|
vcl.bufEnd = 0
|
||||||
@ -94,38 +144,56 @@ proc open*(vcl: var VCardLexer, input: Stream, bufLen = 16384) =
|
|||||||
vcl.skipUtf8Bom
|
vcl.skipUtf8Bom
|
||||||
|
|
||||||
proc setBookmark*(vcl: var VCardLexer) =
|
proc setBookmark*(vcl: var VCardLexer) =
|
||||||
vcl.bookmark = vcl.pos
|
## Set a bookmark into the lexer's buffer. This will prevent the lexer from
|
||||||
vcl.bookmarkVal = newStringOfCap(32)
|
## discarding data from this bookmark forward when it refills.
|
||||||
|
##
|
||||||
|
## The bookmark must be cleared using either `returnToBookmark` or
|
||||||
|
## `unsetBookmark`, otherwise this lexer will no function as a streaming
|
||||||
|
## lexer and will end up reading the entire remainder of the input stream
|
||||||
|
## into memory (if it can).
|
||||||
|
##
|
||||||
|
## This function can be called multiple times in order to create nested
|
||||||
|
## bookmarks. For example, we might set a bookmark at the beginning of a line
|
||||||
|
## to be able to reset if we fail to parse the line, then set a bookmark
|
||||||
|
## midway through when attempting to parse a parameter value. Care should be
|
||||||
|
## taken when nesting bookmarks as all bookmarks must be released to avoid
|
||||||
|
## the behavior described above.
|
||||||
|
vcl.bookmark.add(vcl.pos)
|
||||||
|
vcl.bookmarkVal.add(newStringOfCap(32))
|
||||||
|
|
||||||
proc returnToBookmark*(vcl: var VCardLexer) =
|
proc returnToBookmark*(vcl: var VCardLexer) =
|
||||||
vcl.pos = vcl.bookmark
|
## Unset the most recent bookmark, resetting the lexer's read position to the
|
||||||
vcl.bookmark = -1
|
## position of the bookmark.
|
||||||
|
if vcl.bookmark.len == 0: return
|
||||||
|
vcl.pos = vcl.bookmark.pop()
|
||||||
|
let valRead = vcl.bookmarkVal.pop()
|
||||||
|
for idx in 0..<vcl.bookmarkVal.len:
|
||||||
|
if vcl.bookmarkVal[idx].len > valRead.len:
|
||||||
|
vcl.bookmarkVal[idx] = vcl.bookmarkVal[idx][0 ..< ^valRead.len]
|
||||||
|
|
||||||
proc unsetBookmark*(vcl: var VCardLexer) =
|
proc unsetBookmark*(vcl: var VCardLexer) =
|
||||||
vcl.bookmark = -1
|
## Discard the most recent bookmark, leaving the lexer's read position at its
|
||||||
|
## current position..
|
||||||
|
if vcl.bookmark.len == 0: return
|
||||||
|
discard vcl.bookmark.pop()
|
||||||
|
discard vcl.bookmarkVal.pop()
|
||||||
|
|
||||||
proc readSinceBookmark*(vcl: var VCardLexer): string =
|
proc readSinceBookmark*(vcl: var VCardLexer): string =
|
||||||
return vcl.bookmarkVal
|
## Get the value read since the last bookmark.
|
||||||
#[
|
if vcl.bookmarkVal.len > 0:
|
||||||
if vcl.pos < vcl.bookmark:
|
return vcl.bookmarkVal[^1]
|
||||||
# p e s b
|
else: return ""
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
|
||||||
result = newStringOfCap(vcl.buffer.len - vcl.bookmark + vcl.pos)
|
|
||||||
else:
|
|
||||||
# s b p e
|
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
|
||||||
result = newStringOfCap(vcl.pos - vcl.bookmark)
|
|
||||||
|
|
||||||
let curPos = vcl.pos
|
|
||||||
vcl.pos = vcl.bookmark
|
|
||||||
while vcl.pos != curPos: result.add(vcl.read)
|
|
||||||
]#
|
|
||||||
|
|
||||||
proc isLineWrap(vcl: var VCardLexer, allowRefill = true): bool =
|
proc isLineWrap(vcl: var VCardLexer, allowRefill = true): bool =
|
||||||
|
## Answers the question "is this a folded line"? Transparently handles the
|
||||||
|
## case where it needs to refill the buffer to answer this question.
|
||||||
if vcl.buffer[vcl.pos] != '\r': return false
|
if vcl.buffer[vcl.pos] != '\r': return false
|
||||||
|
|
||||||
# less than three characters in the buffer
|
# less than three characters in the buffer
|
||||||
if wrappedIdx(vcl.pos + 3) > vcl.bufEnd:
|
if wrappedIdx(vcl.pos + 3) > vcl.bufEnd:
|
||||||
|
# Only try to refill the buffer once. If we re-enter and still don't have
|
||||||
|
# three characters, we know we were unable to fill the buffer and are
|
||||||
|
# likely at the end.
|
||||||
if allowRefill:
|
if allowRefill:
|
||||||
vcl.fillBuffer()
|
vcl.fillBuffer()
|
||||||
return vcl.isLineWrap(false)
|
return vcl.isLineWrap(false)
|
||||||
@ -137,54 +205,87 @@ proc isLineWrap(vcl: var VCardLexer, allowRefill = true): bool =
|
|||||||
vcl.buffer[wrappedIdx(vcl.pos + 2)] == ' '
|
vcl.buffer[wrappedIdx(vcl.pos + 2)] == ' '
|
||||||
|
|
||||||
proc read*(vcl: var VCardLexer, peek = false): char =
|
proc read*(vcl: var VCardLexer, peek = false): char =
|
||||||
|
## Read one byte off of the input stream. By default this will advance the
|
||||||
|
## lexer read position by one byte. If `peek` is set to `true`, this will
|
||||||
|
## leave the read position at the same logical position. The underlying
|
||||||
|
## buffer position may still change if, for example, the next byte is the
|
||||||
|
## beginning of a folded line wrap. In this case the internal buffer position
|
||||||
|
## will advance past that line wrap.
|
||||||
|
|
||||||
if vcl.atEnd: vcl.fillBuffer()
|
if vcl.atEnd: vcl.fillBuffer()
|
||||||
|
|
||||||
if vcl.isLineWrap:
|
if vcl.isLineWrap:
|
||||||
vcl.pos += 3
|
vcl.pos += 3
|
||||||
vcl.lineNumber += 1
|
vcl.lineNumber += 1
|
||||||
vcl.lineStart = vcl.pos
|
vcl.lineStart = vcl.pos
|
||||||
|
vcl.lineVal = newStringOfCap(84)
|
||||||
if vcl.atEnd: vcl.fillBuffer()
|
if vcl.atEnd: vcl.fillBuffer()
|
||||||
|
|
||||||
elif vcl.buffer[vcl.pos] == '\n':
|
elif vcl.buffer[vcl.pos] == '\n' and not peek:
|
||||||
vcl.lineNumber += 1
|
vcl.lineNumber += 1
|
||||||
vcl.lineStart = wrappedIdx(vcl.pos + 1)
|
vcl.lineStart = wrappedIdx(vcl.pos + 1)
|
||||||
|
vcl.lineVal = newStringOfCap(84)
|
||||||
|
|
||||||
result = vcl.buffer[vcl.pos]
|
result = vcl.buffer[vcl.pos]
|
||||||
if not peek:
|
if not peek:
|
||||||
if vcl.bookmark != -1: vcl.bookmarkVal.add(result)
|
for idx in 0..<vcl.bookmarkVal.len: vcl.bookmarkVal[idx].add(result)
|
||||||
|
vcl.lineVal.add(result)
|
||||||
vcl.pos = wrappedIdx(vcl.pos + 1)
|
vcl.pos = wrappedIdx(vcl.pos + 1)
|
||||||
|
|
||||||
|
proc readLen*(vcl: var VCardLexer, bytesToRead: int, peek = false): string =
|
||||||
|
## Convenience procedure to read multiple bytes (if able) and return the
|
||||||
|
## value read.
|
||||||
|
result = newStringOfCap(bytesToRead)
|
||||||
|
for i in 0..<bytesToRead: result.add(vcl.read)
|
||||||
|
|
||||||
proc readRune*(vcl: var VCardLexer, peek = false): Rune =
|
proc readRune*(vcl: var VCardLexer, peek = false): Rune =
|
||||||
|
## Read one unicode rune off of the input stream. By default this will
|
||||||
|
## advance the lexer read position by the byte length of the rune read. If
|
||||||
|
## `peek` is set to `true`, this will leave the read position at the same
|
||||||
|
## logical position. The underlying buffer position may still change if, for
|
||||||
|
## example, the next rune is the beginning of a folded line wrap. In this
|
||||||
|
## case the internal buffer position will advance past that line wrap.
|
||||||
|
|
||||||
if vcl.atEnd: vcl.fillBuffer()
|
if vcl.atEnd: vcl.fillBuffer()
|
||||||
|
|
||||||
if vcl.isLineWrap:
|
if vcl.isLineWrap:
|
||||||
vcl.pos += 3
|
vcl.pos += 3
|
||||||
vcl.lineNumber += 1
|
vcl.lineNumber += 1
|
||||||
vcl.lineStart = vcl.pos
|
vcl.lineStart = vcl.pos
|
||||||
|
vcl.lineVal = newStringOfCap(84)
|
||||||
if vcl.atEnd: vcl.fillBuffer()
|
if vcl.atEnd: vcl.fillBuffer()
|
||||||
|
|
||||||
elif vcl.buffer[vcl.pos] == '\n':
|
elif vcl.buffer[vcl.pos] == '\n':
|
||||||
vcl.lineNumber += 1
|
vcl.lineNumber += 1
|
||||||
vcl.lineStart = wrappedIdx(vcl.pos + 1)
|
vcl.lineStart = wrappedIdx(vcl.pos + 1)
|
||||||
|
vcl.lineVal = newStringOfCap(84)
|
||||||
|
|
||||||
result = vcl.buffer.runeAt(vcl.pos)
|
result = vcl.buffer.runeAt(vcl.pos)
|
||||||
if not peek: vcl.pos += vcl.buffer.runeLenAt(vcl.pos)
|
if not peek:
|
||||||
|
for idx in 0..<vcl.bookmarkVal.len: vcl.bookmarkVal[idx].add(result)
|
||||||
|
vcl.lineVal.add(result)
|
||||||
|
vcl.pos += vcl.buffer.runeLenAt(vcl.pos)
|
||||||
|
|
||||||
|
proc readRunesLen*(vcl: var VCardLexer, runesToRead: int, peek = false): string =
|
||||||
|
## Convenience procedure to read multiple runes (if able) and return the
|
||||||
|
## value read.
|
||||||
|
result = newStringOfCap(runesToRead * 4)
|
||||||
|
for i in 0..<runesToRead: result.add(vcl.readRune)
|
||||||
|
|
||||||
proc peek*(vcl: var VCardLexer): char =
|
proc peek*(vcl: var VCardLexer): char =
|
||||||
|
## Convenience method to call `read(peek = true)`
|
||||||
return vcl.read(peek = true)
|
return vcl.read(peek = true)
|
||||||
|
|
||||||
proc peekRune*(vcl: var VCardLexer): Rune =
|
proc peekRune*(vcl: var VCardLexer): Rune =
|
||||||
|
## Convenience method to call `read(peek = true)`
|
||||||
return vcl.readRune(peek = true)
|
return vcl.readRune(peek = true)
|
||||||
|
|
||||||
proc getColNumber*(vcl: VCardLexer, pos: int): int =
|
proc getColNumber*(vcl: VCardLexer, pos: int): int =
|
||||||
|
## Calculate the column number of the lexer's current read position relative
|
||||||
|
## to the start of the most recent line.
|
||||||
if vcl.lineStart < pos: return pos - vcl.lineStart
|
if vcl.lineStart < pos: return pos - vcl.lineStart
|
||||||
else: return (vcl.buffer.len - vcl.lineStart) + pos
|
else: return (vcl.buffer.len - vcl.lineStart) + pos
|
||||||
|
|
||||||
## Unit Tests
|
|
||||||
## ============================================================================
|
|
||||||
|
|
||||||
import std/unittest
|
|
||||||
|
|
||||||
proc dumpLexerState*(l: VCardLexer): string =
|
proc dumpLexerState*(l: VCardLexer): string =
|
||||||
result =
|
result =
|
||||||
"pos = " & $l.pos & "\p" &
|
"pos = " & $l.pos & "\p" &
|
||||||
@ -195,7 +296,9 @@ proc dumpLexerState*(l: VCardLexer): string =
|
|||||||
"bufEnd = " & $l.bufEnd & "\p" &
|
"bufEnd = " & $l.bufEnd & "\p" &
|
||||||
"buffer = " & l.buffer & "\p"
|
"buffer = " & l.buffer & "\p"
|
||||||
|
|
||||||
suite "vcard/lexer":
|
## Unit Tests
|
||||||
|
## ============================================================================
|
||||||
|
proc runVcardLexerPrivateTests*() =
|
||||||
|
|
||||||
const longTestString =
|
const longTestString =
|
||||||
"This is my test string. There are many like it but this one is mine."
|
"This is my test string. There are many like it but this one is mine."
|
||||||
@ -212,143 +315,204 @@ suite "vcard/lexer":
|
|||||||
return false
|
return false
|
||||||
return true
|
return true
|
||||||
|
|
||||||
#test "fillBuffer doesn't double the buffer needlessly":
|
|
||||||
# var l: VCardLexer
|
|
||||||
|
|
||||||
proc readExpected(vcl: var VCardLexer, s: string): bool =
|
proc readExpected(vcl: var VCardLexer, s: string): bool =
|
||||||
for i in 0..<s.len:
|
for i in 0..<s.len:
|
||||||
if vcl.read != s[i]:
|
if vcl.read != s[i]:
|
||||||
return false
|
return false
|
||||||
return true
|
return true
|
||||||
|
|
||||||
test "can open and fill buffer":
|
# "can open and fill buffer":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream("test"))
|
l.open(newStringStream("test"))
|
||||||
check:
|
assert l.bufferIs("test")
|
||||||
l.bufferIs("test")
|
assert not l.isFull
|
||||||
not l.isFull
|
assert l.readExpected("test")
|
||||||
l.readExpected("test")
|
|
||||||
|
|
||||||
test "refills buffer when emptied":
|
# "refills buffer when emptied":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream("test"), 3)
|
l.open(newStringStream("test"), 3)
|
||||||
check:
|
assert l.bufferIs("te")
|
||||||
l.bufferIs("te")
|
assert l.isFull
|
||||||
l.isFull
|
assert l.read == 't'
|
||||||
l.read == 't'
|
assert l.read == 'e'
|
||||||
l.read == 'e'
|
assert l.read == 's'
|
||||||
l.read == 's'
|
assert l.bufferIs("st")
|
||||||
l.bufferIs("st")
|
assert l.read == 't'
|
||||||
l.read == 't'
|
|
||||||
|
|
||||||
test "isFull correctness":
|
# "isFull correctness":
|
||||||
|
block:
|
||||||
var l = VCardLexer(
|
var l = VCardLexer(
|
||||||
pos: 0,
|
pos: 0,
|
||||||
bookmark: -1,
|
bookmark: @[],
|
||||||
buffer: "0123456789",
|
buffer: "0123456789",
|
||||||
bufStart: 0,
|
bufStart: 0,
|
||||||
bufEnd: 9)
|
bufEnd: 9)
|
||||||
|
|
||||||
# s e
|
# s e
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
check l.isFull
|
assert l.isFull
|
||||||
|
|
||||||
# s p e
|
# s p e
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
discard l.read
|
discard l.read
|
||||||
check not l.isFull
|
assert not l.isFull
|
||||||
|
|
||||||
# e s
|
# e s
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
l.bufStart = 3
|
l.bufStart = 3
|
||||||
l.pos = 3
|
l.pos = 3
|
||||||
l.bufEnd = 2
|
l.bufEnd = 2
|
||||||
check l.isFull
|
assert l.isFull
|
||||||
|
|
||||||
# e s p
|
# e s p
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
discard l.read
|
discard l.read
|
||||||
check:
|
assert l.pos == 4
|
||||||
l.pos == 4
|
assert not l.isFull
|
||||||
not l.isFull
|
|
||||||
|
|
||||||
# e s
|
# e s
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
l.bufStart = 9
|
l.bufStart = 9
|
||||||
l.pos = 9
|
l.pos = 9
|
||||||
l.bufEnd = 8
|
l.bufEnd = 8
|
||||||
check l.isFull
|
assert l.isFull
|
||||||
|
|
||||||
# p e s
|
# p e s
|
||||||
# 0 1 2 3 4 5 6 7 8 9
|
# 0 1 2 3 4 5 6 7 8 9
|
||||||
discard l.read
|
discard l.read
|
||||||
check:
|
assert l.pos == 0
|
||||||
l.pos == 0
|
assert not l.isFull
|
||||||
not l.isFull
|
|
||||||
|
|
||||||
test "handles wrapped lines":
|
# "handles wrapped lines":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream("line\r\n wrap\r\nline 2"), 3)
|
l.open(newStringStream("line\r\n wrap\r\nline 2"), 3)
|
||||||
|
|
||||||
check l.readExpected("line wrap\r\nline 2")
|
assert l.readExpected("line wrap\r\nline 2")
|
||||||
|
|
||||||
test "fillBuffer correctness":
|
# "fillBuffer correctness":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream(longTestString), 5)
|
l.open(newStringStream(longTestString), 5)
|
||||||
check:
|
assert l.bufferIs(longTestString[0..<4])
|
||||||
l.bufferIs(longTestString[0..<4])
|
assert l.isFull
|
||||||
l.isFull
|
assert l.bufStart == 0
|
||||||
l.bufStart == 0
|
assert l.bufEnd == 4
|
||||||
l.bufEnd == 4
|
assert l.pos == 0
|
||||||
l.pos == 0
|
assert l.readExpected("Th")
|
||||||
l.readExpected("Th")
|
assert not l.isFull
|
||||||
not l.isFull
|
assert not l.atEnd
|
||||||
not l.atEnd
|
assert l.pos == 2
|
||||||
l.pos == 2
|
|
||||||
|
|
||||||
l.fillBuffer
|
l.fillBuffer
|
||||||
check:
|
assert l.isFull
|
||||||
l.isFull
|
assert l.bufEnd == 1
|
||||||
l.bufEnd == 1
|
assert l.pos == 2
|
||||||
l.pos == 2
|
assert l.bufStart == 2
|
||||||
l.bufStart == 2
|
|
||||||
|
|
||||||
test "bookmark preserves the buffer":
|
# "bookmark preserves the buffer":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream(longTestString), 7)
|
l.open(newStringStream(longTestString), 7)
|
||||||
check:
|
assert l.buffer.len == 7
|
||||||
l.buffer.len == 7
|
assert l.bufferIs(longTestString[0..<6])
|
||||||
l.bufferIs(longTestString[0..<6])
|
assert l.isFull
|
||||||
l.isFull
|
assert l.bufEnd == 6
|
||||||
l.bufEnd == 6
|
assert l.pos == 0
|
||||||
l.pos == 0
|
assert l.bookmark == @[]
|
||||||
l.bookmark == -1
|
assert l.readExpected(longTestString[0..<5])
|
||||||
l.readExpected(longTestString[0..<5])
|
assert not l.isFull
|
||||||
not l.isFull
|
assert not l.atEnd
|
||||||
not l.atEnd
|
assert l.pos == 5
|
||||||
l.pos == 5
|
|
||||||
|
|
||||||
l.setBookmark
|
l.setBookmark
|
||||||
# read enough to require us to refill the buffer.
|
# read enough to require us to refill the buffer.
|
||||||
check:
|
assert l.bookmark == @[5]
|
||||||
l.bookmark == 5
|
assert l.readExpected(longTestString[5..<10])
|
||||||
l.readExpected(longTestString[5..<10])
|
assert l.pos == 3
|
||||||
l.pos == 3
|
assert newStartIdx(l) == 5
|
||||||
newStartIdx(l) == 5
|
assert l.buffer.len == 7
|
||||||
l.buffer.len == 7
|
|
||||||
|
|
||||||
l.returnToBookmark
|
l.returnToBookmark
|
||||||
check:
|
assert l.bookmark == @[]
|
||||||
l.bookmark == -1
|
assert l.pos == 5
|
||||||
l.pos == 5
|
|
||||||
|
|
||||||
test "readRune":
|
# "can set and unset multiple bookmarks"
|
||||||
|
block:
|
||||||
|
var l: VCardLexer
|
||||||
|
l.open(newStringStream(longTestString))
|
||||||
|
assert l.pos == 0
|
||||||
|
assert l.bookmark == @[]
|
||||||
|
assert l.readExpected("This is my ")
|
||||||
|
|
||||||
|
l.setBookmark
|
||||||
|
assert l.bookmark == @[11]
|
||||||
|
assert l.bookmarkVal == @[""]
|
||||||
|
|
||||||
|
assert l.readExpected("test string")
|
||||||
|
assert l.bookmark == @[11]
|
||||||
|
assert l.bookmarkVal == @["test string"]
|
||||||
|
assert l.readSinceBookmark == "test string"
|
||||||
|
|
||||||
|
l.setBookmark
|
||||||
|
assert l.bookmark == @[11, 22]
|
||||||
|
assert l.bookmarkVal == @["test string", ""]
|
||||||
|
|
||||||
|
assert l.readExpected(". There are many")
|
||||||
|
assert l.bookmarkVal == @["test string. There are many", ". There are many"]
|
||||||
|
assert l.readSinceBookmark == ". There are many"
|
||||||
|
assert l.pos == 38
|
||||||
|
|
||||||
|
l.unsetBookmark
|
||||||
|
assert l.pos == 38
|
||||||
|
assert l.bookmark == @[11]
|
||||||
|
assert l.bookmarkVal == @["test string. There are many"]
|
||||||
|
assert l.readSinceBookmark == "test string. There are many"
|
||||||
|
|
||||||
|
l.unsetBookmark
|
||||||
|
assert l.pos == 38
|
||||||
|
assert l.bookmark == @[]
|
||||||
|
assert l.bookmarkVal == @[]
|
||||||
|
assert l.readSinceBookmark == ""
|
||||||
|
|
||||||
|
# "can set and return to multiple bookmarks"
|
||||||
|
block:
|
||||||
|
var l: VCardLexer
|
||||||
|
l.open(newStringStream(longTestString))
|
||||||
|
assert l.pos == 0
|
||||||
|
assert l.bookmark == @[]
|
||||||
|
assert l.readExpected("This is my ")
|
||||||
|
|
||||||
|
l.setBookmark
|
||||||
|
assert l.readExpected("test string")
|
||||||
|
l.setBookmark
|
||||||
|
assert l.bookmark == @[11, 22]
|
||||||
|
assert l.readExpected(". There are many")
|
||||||
|
assert l.bookmarkVal == @["test string. There are many", ". There are many"]
|
||||||
|
assert l.pos == 38
|
||||||
|
|
||||||
|
l.returnToBookmark
|
||||||
|
assert l.pos == 22
|
||||||
|
assert l.bookmark == @[11]
|
||||||
|
assert l.bookmarkVal == @["test string"]
|
||||||
|
assert l.readSinceBookmark == "test string"
|
||||||
|
|
||||||
|
l.returnToBookmark
|
||||||
|
assert l.pos == 11
|
||||||
|
assert l.bookmark == @[]
|
||||||
|
assert l.bookmarkVal == @[]
|
||||||
|
|
||||||
|
# "readRune":
|
||||||
|
block:
|
||||||
var l: VCardLexer
|
var l: VCardLexer
|
||||||
l.open(newStringStream("TEST"))
|
l.open(newStringStream("TEST"))
|
||||||
check:
|
assert l.bufferIs("TEST")
|
||||||
l.bufferIs("TEST")
|
assert l.peekRune == Rune('T')
|
||||||
l.peekRune == Rune('T')
|
assert l.readRune == Rune('T')
|
||||||
l.readRune == Rune('T')
|
assert l.readRune == Rune('E')
|
||||||
l.readRune == Rune('E')
|
assert l.readRune == Rune('S')
|
||||||
l.readRune == Rune('S')
|
assert l.readRune == Rune('T')
|
||||||
l.readRune == Rune('T')
|
|
||||||
|
when isMainModule: runVcardLexerPrivateTests()
|
||||||
|
@ -1,40 +0,0 @@
|
|||||||
import options, strutils
|
|
||||||
import ./lexer
|
|
||||||
|
|
||||||
const WSP* = {' ', '\t'}
|
|
||||||
const ALPHA_NUM* = { 'a'..'z', 'A'..'Z', '0'..'9' }
|
|
||||||
|
|
||||||
proc expect*[T](p: var T, expected: string, caseSensitive = false) =
|
|
||||||
p.setBookmark
|
|
||||||
|
|
||||||
if caseSensitive:
|
|
||||||
for ch in expected:
|
|
||||||
if p.read != ch:
|
|
||||||
p.error("expected '$1' but found '$2'" %
|
|
||||||
[expected, p.readSinceBookmark])
|
|
||||||
|
|
||||||
else:
|
|
||||||
for rune in expected.runes:
|
|
||||||
if p.readRune.toLower != rune.toLower:
|
|
||||||
p.error("expected '$1' but found '$2'" %
|
|
||||||
[ expected, p.readSinceBookmark ])
|
|
||||||
|
|
||||||
p.unsetBookmark
|
|
||||||
|
|
||||||
proc readGroup*[T](p: var T): Option[string] =
|
|
||||||
## All VCARD content items can be optionally prefixed with a group name. This
|
|
||||||
## scans the input to see if there is a group defined at the current read
|
|
||||||
## location. If there is a valid group, the group name is returned and the
|
|
||||||
## read position is advanced past the '.' to the start of the content type
|
|
||||||
## name. If there is not a valid group the read position is left unchanged.
|
|
||||||
|
|
||||||
p.setBookmark
|
|
||||||
var ch = p.read
|
|
||||||
while ALPHA_NUM.contains(ch): ch = p.read
|
|
||||||
|
|
||||||
if (ch == '.'):
|
|
||||||
result = some(readSinceBookmark(p)[0..^2])
|
|
||||||
p.unsetBookmark
|
|
||||||
else:
|
|
||||||
result = none[string]()
|
|
||||||
p.returnToBookmark
|
|
@ -1,55 +0,0 @@
|
|||||||
import sequtils, strutils, times
|
|
||||||
|
|
||||||
const DATE_FMTS = [ "yyyy-MM-dd", "yyyyMMdd" ]
|
|
||||||
const DATE_TIME_FMTS = [
|
|
||||||
"yyyyMMdd'T'HHmmss",
|
|
||||||
"yyyyMMdd'T'HHmmssz",
|
|
||||||
"yyyyMMdd'T'HHmmsszzz",
|
|
||||||
"yyyyMMdd'T'HHmmss'.'fffzzz",
|
|
||||||
"yyyy-MM-dd'T'HH:mm:ss",
|
|
||||||
"yyyy-MM-dd'T'HH:mm:ssz",
|
|
||||||
"yyyy-MM-dd'T'HH:mm:sszzz",
|
|
||||||
"yyyy-MM-dd'T'HH:mm:ss'.'fffzzz",
|
|
||||||
]
|
|
||||||
|
|
||||||
const ALL_FMTS = DATE_TIME_FMTS.toSeq & DATE_FMTS.toSeq
|
|
||||||
|
|
||||||
proc parseDateTimeStr(
|
|
||||||
dateStr: string,
|
|
||||||
dateFmts: openarray[string]
|
|
||||||
): DateTime {.inline, raises:[ValueError].} =
|
|
||||||
|
|
||||||
for fmt in dateFmts:
|
|
||||||
try: result = parse(dateStr, fmt)
|
|
||||||
except ValueError: discard
|
|
||||||
|
|
||||||
if not result.isInitialized:
|
|
||||||
raise newException(ValueError, "cannot parse date: " & dateStr )
|
|
||||||
|
|
||||||
proc parseDate*(dateStr: string): DateTime =
|
|
||||||
parseDateTimeStr(dateStr, DATE_FMTS)
|
|
||||||
|
|
||||||
proc parseDateTime*(dateStr: string): DateTime =
|
|
||||||
parseDateTimeStr(dateStr, DATE_TIME_FMTS)
|
|
||||||
|
|
||||||
proc parseDateOrDateTime*(dateStr: string): DateTime =
|
|
||||||
parseDateTimeStr(dateStr, ALL_FMTS)
|
|
||||||
|
|
||||||
func foldContentLine*(s: string): string =
|
|
||||||
result = ""
|
|
||||||
var rem = s
|
|
||||||
while rem.len > 75: # TODO: unicode multi-byte safe?
|
|
||||||
result &= rem[0..<75] & "\r\n "
|
|
||||||
rem = rem[75..^1]
|
|
||||||
result &= rem
|
|
||||||
|
|
||||||
func unfoldContentLine*(s: string): string =
|
|
||||||
return s.multiReplace([("\r\n ", "")])
|
|
||||||
|
|
||||||
template indexOfIt*(s, pred: untyped): int =
|
|
||||||
var result = -1
|
|
||||||
for idx, it {.inject.} in pairs(s):
|
|
||||||
if pred:
|
|
||||||
result = idx
|
|
||||||
break
|
|
||||||
result
|
|
1412
src/vcard/vcard3.nim
Normal file
1412
src/vcard/vcard3.nim
Normal file
File diff suppressed because it is too large
Load Diff
1491
src/vcard/vcard4.nim
Normal file
1491
src/vcard/vcard4.nim
Normal file
File diff suppressed because it is too large
Load Diff
2030
src/vcard3.nim
2030
src/vcard3.nim
File diff suppressed because it is too large
Load Diff
1
tests/allen.foster.jpg.uri
Normal file
1
tests/allen.foster.jpg.uri
Normal file
File diff suppressed because one or more lines are too long
212
tests/allen.foster.v4.vcf
Normal file
212
tests/allen.foster.v4.vcf
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
BEGIN:VCARD
|
||||||
|
VERSION:4.0
|
||||||
|
PRODID:+//IDN bitfire.at//DAVx5/4.1.1-gplay ez-vcard/0.11.3
|
||||||
|
UID:b7047a2e-c46b-47cb-af0b-94d354b7746a
|
||||||
|
FN:Dr. Allen Foster
|
||||||
|
N;SORT-AS=Foster,Allen,,,:Foster;Jack;John,Allen;Dr.;II
|
||||||
|
NICKNAME:Jack Jr.
|
||||||
|
NICKNAME;TYPE=work;PREF=1:Doc A
|
||||||
|
TEL;TYPE=cell:+1 555-123-4567
|
||||||
|
TEL;TYPE=cell:(555) 123-4567
|
||||||
|
TEL;TYPE=work,voice;VALUE=uri:tel:+1-555-874-1234
|
||||||
|
EMAIL;TYPE=work;PREF=2:jack.foster@company.test
|
||||||
|
EMAIL;TYPE=home;PREF=1:allen@fosters.test
|
||||||
|
SOURCE;VALUE=uri:https://carddav.fosters.test/allen.vcf
|
||||||
|
KIND:individual
|
||||||
|
REV:20220226T060828Z
|
||||||
|
BDAY;ALTID=1;VALUE=date-and-or-time:--1224
|
||||||
|
BDAY;ALTID=1;VALUE=text:Christmas Eve
|
||||||
|
ANNIVERSARY:20140612T163000-0500
|
||||||
|
GENDER:M;male
|
||||||
|
MADE-UP-PROP:Sample value for my made-up prop.
|
||||||
|
NOTE;LANG=en-us:This is an example\, for clarity; in text value cases the parser
|
||||||
|
will recognize escape values for '\,'\, '\\'\, and newlines. For example:\n 12
|
||||||
|
3 Flagstaff Road\N Placeville\, MA
|
||||||
|
X-CUSTOM-EXAMPLE;PARAM="How one says, ^'Hello.^'";LABEL=^^top^nsecond line:This
|
||||||
|
is an example, for clarity; in straight value cases, the parser does not reco
|
||||||
|
gnize any escape values, as the meaning of the content is implementation-speci
|
||||||
|
fic.
|
||||||
|
PHOTO;ALTID=1;VALUE=uri:https://tile.loc.gov/storage-services/service/pnp/
|
||||||
|
bellcm/02200/02297r.jpg
|
||||||
|
URL:https://allen.fosters.test/
|
||||||
|
PHOTO;ALTID=1:
|
||||||
|
kqAAgAAAAHABIBAwABAAAAAQAAABoBBQABAAAAYgAAABsBBQABAAAAagAAACgBAwABAAAAAgAAADEB
|
||||||
|
AgANAAAAcgAAADIBAgAUAAAAgAAAAGmHBAABAAAAlAAAAAAAAAB4BQAAAQAAAHgFAAABAAAAR0lNUC
|
||||||
|
AyLjEwLjM0AAAyMDIzOjA0OjI1IDE2OjQzOjUyAAEAAaADAAEAAAABAAAAAAAAAP/hDM1odHRwOi8v
|
||||||
|
bnMuYWRvYmUuY29tL3hhcC8xLjAvADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaU
|
||||||
|
h6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1w
|
||||||
|
dGs9IlhNUCBDb3JlIDQuNC4wLUV4aXYyIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3Ln
|
||||||
|
czLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJv
|
||||||
|
dXQ9IiIgeG1sbnM6eG1wTU09Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9tbS8iIHhtbG5zOn
|
||||||
|
N0RXZ0PSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvc1R5cGUvUmVzb3VyY2VFdmVudCMiIHht
|
||||||
|
bG5zOmRjPSJodHRwOi8vcHVybC5vcmcvZGMvZWxlbWVudHMvMS4xLyIgeG1sbnM6R0lNUD0iaHR0cD
|
||||||
|
ovL3d3dy5naW1wLm9yZy94bXAvIiB4bWxuczp4bXA9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEu
|
||||||
|
MC8iIHhtcE1NOkRvY3VtZW50SUQ9ImdpbXA6ZG9jaWQ6Z2ltcDpmYzVkZDFkMC05ZmNiLTRhZjAtOG
|
||||||
|
UzNS1jMTMyMDU4NTUwMmEiIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6YmEyOThkOGYtMDY3NC00
|
||||||
|
ZDgzLWJhZGMtNWVkY2Y2OTg2NTBjIiB4bXBNTTpPcmlnaW5hbERvY3VtZW50SUQ9InhtcC5kaWQ6Nz
|
||||||
|
JjYWViYjMtMjkzMy00ZGJmLTg0M2EtYzYwYjBkZWYzMzdlIiBkYzpGb3JtYXQ9ImltYWdlL2pwZWci
|
||||||
|
IEdJTVA6QVBJPSIyLjAiIEdJTVA6UGxhdGZvcm09IldpbmRvd3MiIEdJTVA6VGltZVN0YW1wPSIxNj
|
||||||
|
gyNDU5MDQ2MjA4NjE3IiBHSU1QOlZlcnNpb249IjIuMTAuMzQiIHhtcDpDcmVhdG9yVG9vbD0iR0lN
|
||||||
|
UCAyLjEwIiB4bXA6TWV0YWRhdGFEYXRlPSIyMDIzOjA0OjI1VDE2OjQzOjUyLTA1OjAwIiB4bXA6TW
|
||||||
|
9kaWZ5RGF0ZT0iMjAyMzowNDoyNVQxNjo0Mzo1Mi0wNTowMCI+IDx4bXBNTTpIaXN0b3J5PiA8cmRm
|
||||||
|
OlNlcT4gPHJkZjpsaSBzdEV2dDphY3Rpb249InNhdmVkIiBzdEV2dDpjaGFuZ2VkPSIvIiBzdEV2dD
|
||||||
|
ppbnN0YW5jZUlEPSJ4bXAuaWlkOjU2YzcxYmUyLWExNmMtNDE2OC1iNDA5LWI3YjRlMTgwZTFmMyIg
|
||||||
|
c3RFdnQ6c29mdHdhcmVBZ2VudD0iR2ltcCAyLjEwIChXaW5kb3dzKSIgc3RFdnQ6d2hlbj0iMjAyMy
|
||||||
|
0wNC0yNVQxNjo0NDowNiIvPiA8L3JkZjpTZXE+IDwveG1wTU06SGlzdG9yeT4gPC9yZGY6RGVzY3Jp
|
||||||
|
cHRpb24+IDwvcmRmOlJERj4gPC94OnhtcG1ldGE+ICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAg
|
||||||
|
ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC
|
||||||
|
AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPD94cGFja2V0IGVuZD0idyI/Pv/iAjBJQ0Nf
|
||||||
|
UFJPRklMRQABAQAAAiBsY21zBEAAAG1udHJHUkFZWFlaIAfnAAQAGQAVACoAJWFjc3BNU0ZUAAAAAA
|
||||||
|
AAAAAAAAAAAAAAAAAAAAAAAAAAAAD21gABAAAAANMtbGNtcwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
|
||||||
|
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABmRlc2MAAADMAAAAbmNwcnQAAAE8AAAANnd0cHQAAAF0AA
|
||||||
|
AAFGtUUkMAAAGIAAAAIGRtbmQAAAGoAAAAJGRtZGQAAAHMAAAAUm1sdWMAAAAAAAAAAQAAAAxlblVT
|
||||||
|
AAAAUgAAABwARwBJAE0AUAAgAGIAdQBpAGwAdAAtAGkAbgAgAEQANgA1ACAARwByAGEAeQBzAGMAYQ
|
||||||
|
BsAGUAIAB3AGkAdABoACAAcwBSAEcAQgAgAFQAUgBDAABtbHVjAAAAAAAAAAEAAAAMZW5VUwAAABoA
|
||||||
|
AAAcAFAAdQBiAGwAaQBjACAARABvAG0AYQBpAG4AAFhZWiAAAAAAAADzUQABAAAAARbMcGFyYQAAAA
|
||||||
|
AAAwAAAAJmZgAA8qcAAA1ZAAAT0AAAClttbHVjAAAAAAAAAAEAAAAMZW5VUwAAAAgAAAAcAEcASQBN
|
||||||
|
AFBtbHVjAAAAAAAAAAEAAAAMZW5VUwAAADYAAAAcAEQANgA1ACAARwByAGEAeQBzAGMAYQBsAGUAIA
|
||||||
|
B3AGkAdABoACAAcwBSAEcAQgAgAFQAUgBDAAD/2wBDAAoHBwgHBgoICAgLCgoLDhgQDg0NDh0VFhEY
|
||||||
|
Ix8lJCIfIiEmKzcvJik0KSEiMEExNDk7Pj4+JS5ESUM8SDc9Pjv/wgALCAEVAMgBAREA/8QAGwAAAQ
|
||||||
|
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
|
||||||
|
GAaDKL5DfxBTTUUbUr1Zb/APP/AP/Z
|
||||||
|
END:VCARD
|
@ -1,6 +1,6 @@
|
|||||||
BEGIN:VCARD
|
BEGIN:VCARD
|
||||||
PRODID:-//CyrusIMAP.org//Cyrus 3.7.0-alpha0-927-gf4c98c8499-fm-202208..//EN
|
|
||||||
VERSION:3.0
|
VERSION:3.0
|
||||||
|
PRODID:-//CyrusIMAP.org//Cyrus 3.7.0-alpha0-927-gf4c98c8499-fm-202208..//EN
|
||||||
UID:cdaf67dc-b702-41ac-9c26-bb61df3032d2
|
UID:cdaf67dc-b702-41ac-9c26-bb61df3032d2
|
||||||
N:Bernard;Jonathan;;;
|
N:Bernard;Jonathan;;;
|
||||||
FN:Jonathan Bernard
|
FN:Jonathan Bernard
|
||||||
|
@ -1,2 +1,6 @@
|
|||||||
import unittest
|
import unittest
|
||||||
import vcard/private/lexer
|
import ./vcard/private/lexer
|
||||||
|
|
||||||
|
suite "vcard/private/lexer":
|
||||||
|
test "private lexer tests":
|
||||||
|
runVcardLexerPrivateTests()
|
||||||
|
@ -1,24 +1,42 @@
|
|||||||
import options, unittest, vcard3, zero_functional
|
import options, unittest, zero_functional
|
||||||
|
|
||||||
|
import ./vcard
|
||||||
|
import ./vcard/vcard3
|
||||||
|
|
||||||
suite "vcard/vcard3":
|
suite "vcard/vcard3":
|
||||||
|
|
||||||
let testVCard =
|
test "vcard3/private tests":
|
||||||
"BEGIN:VCARD\r\n" &
|
runVcard3PrivateTests()
|
||||||
"VERSION:3.0\r\n" &
|
|
||||||
"FN:Mr. John Q. Public\\, Esq.\r\n" &
|
|
||||||
"N:Public;John;Quinlan;Mr.;Esq.\r\n" &
|
|
||||||
"END:VCARD\r\n"
|
|
||||||
|
|
||||||
test "minimal VCard":
|
let jdbVCard = readFile("tests/jdb.vcf")
|
||||||
let vc = parseVCard3(testVCard)[0]
|
|
||||||
|
test "parseVCard3":
|
||||||
|
check parseVCards(jdbVCard).len == 1
|
||||||
|
|
||||||
|
test "parseVCard3File":
|
||||||
|
check parseVCardsFromFile("tests/jdb.vcf").len == 1
|
||||||
|
|
||||||
|
# TODO: remove cast after finishing VCard4 implementation
|
||||||
|
let jdb = cast[VCard3](parseVCards(jdbVCard)[0])
|
||||||
|
|
||||||
|
test "email is parsed correctly":
|
||||||
check:
|
check:
|
||||||
vc.n.family[0] == "Public"
|
jdb.email.len == 7
|
||||||
vc.n.given[0] == "John"
|
jdb.email[0].value == "jonathan@jdbernard.com"
|
||||||
vc.fn.value == "Mr. John Q. Public\\, Esq."
|
jdb.email[0].emailType.contains("pref")
|
||||||
|
jdb.email[0].emailType.contains("home")
|
||||||
|
jdb.email[1].value == "jdb@jdb-software.com"
|
||||||
|
jdb.email[1].emailType.contains("work")
|
||||||
|
jdb.email[2].group.isSome
|
||||||
|
jdb.email[2].group.get == "email2"
|
||||||
|
jdb.email[6].value == "jbernard@vectra.ai"
|
||||||
|
jdb.email[6].emailType.contains("work")
|
||||||
|
|
||||||
test "serialize minimal VCard":
|
test "tel is parsed correctly":
|
||||||
let vc = parseVCard3(testVCard)[0]
|
check:
|
||||||
check $vc == testVCard
|
jdb.tel.len == 2
|
||||||
|
jdb.tel[0].value == "(512) 777-1602"
|
||||||
|
jdb.tel[0].telType.contains("CELL")
|
||||||
|
|
||||||
test "RFC2426 Author's VCards":
|
test "RFC2426 Author's VCards":
|
||||||
let vcardsStr =
|
let vcardsStr =
|
||||||
@ -47,18 +65,9 @@ suite "vcard/vcard3":
|
|||||||
"EMAIL;TYPE=INTERNET:howes@netscape.com\r\n" &
|
"EMAIL;TYPE=INTERNET:howes@netscape.com\r\n" &
|
||||||
"END:vCard\r\n"
|
"END:vCard\r\n"
|
||||||
|
|
||||||
let vcards = parseVCard3(vcardsStr)
|
let vcards = parseVCards(vcardsStr)
|
||||||
check:
|
check:
|
||||||
vcards.len == 2
|
vcards.len == 2
|
||||||
vcards[0].fn.value == "Frank Dawson"
|
cast[VCard3](vcards[0]).fn.value == "Frank Dawson"
|
||||||
vcards[0].email.len == 2
|
cast[VCard3](vcards[0]).email.len == 2
|
||||||
(vcards[0].email --> find(it.emailType.contains("PREF"))).isSome
|
(cast[VCard3](vcards[0]).email --> find(it.emailType.contains("PREF"))).isSome
|
||||||
|
|
||||||
test "Jonathan Bernard VCard":
|
|
||||||
#const jdbVcard = readFile("tests/jdb.vcf")
|
|
||||||
let jdb = parseVCard3File("tests/jdb.vcf")[0]
|
|
||||||
check:
|
|
||||||
jdb.email.len == 7
|
|
||||||
jdb.email[0].value == "jonathan@jdbernard.com"
|
|
||||||
jdb.email[0].emailType.contains("pref")
|
|
||||||
jdb.fn.value == "Jonathan Bernard"
|
|
||||||
|
250
tests/tvcard4.nim
Normal file
250
tests/tvcard4.nim
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
import std/[options, strutils, tables, unittest]
|
||||||
|
import zero_functional
|
||||||
|
|
||||||
|
import ./vcard
|
||||||
|
import ./vcard/vcard4
|
||||||
|
|
||||||
|
suite "vcard/vcard4":
|
||||||
|
|
||||||
|
test "vcard4/private tests":
|
||||||
|
runVcard4PrivateTests()
|
||||||
|
|
||||||
|
let v4ExampleStr = readFile("tests/allen.foster.v4.vcf")
|
||||||
|
|
||||||
|
let testVCardTemplate =
|
||||||
|
"BEGIN:VCARD\r\n" &
|
||||||
|
"VERSION:4.0\r\n" &
|
||||||
|
"$#" &
|
||||||
|
"END:VCARD\r\n"
|
||||||
|
|
||||||
|
test "parseVCard4":
|
||||||
|
check parseVCards(v4ExampleStr).len == 1
|
||||||
|
|
||||||
|
test "parseVCard4File":
|
||||||
|
check parseVCardsFromFile("tests/allen.foster.v4.vcf").len == 1
|
||||||
|
|
||||||
|
# TODO: remove cast after finishing VCard4 implementation
|
||||||
|
let v4Ex = cast[VCard4](parseVCards(v4ExampleStr)[0])
|
||||||
|
|
||||||
|
test "RFC 6350 author's VCard":
|
||||||
|
let vcardStr =
|
||||||
|
"BEGIN:VCARD\r\n" &
|
||||||
|
"VERSION:4.0\r\n" &
|
||||||
|
"FN:Simon Perreault\r\n" &
|
||||||
|
"N:Perreault;Simon;;;ing. jr,M.Sc.\r\n" &
|
||||||
|
"BDAY:--0203\r\n" &
|
||||||
|
"ANNIVERSARY:20090808T1430-0500\r\n" &
|
||||||
|
"GENDER:M\r\n" &
|
||||||
|
"LANG;PREF=1:fr\r\n" &
|
||||||
|
"LANG;PREF=2:en\r\n" &
|
||||||
|
"ORG;TYPE=work:Viagenie\r\n" &
|
||||||
|
"ADR;TYPE=work:;Suite D2-630;2875 Laurier;\r\n" &
|
||||||
|
" Quebec;QC;G1V 2M2;Canada\r\n" &
|
||||||
|
"TEL;VALUE=uri;TYPE=\"work,voice\";PREF=1:tel:+1-418-656-9254;ext=102\r\n" &
|
||||||
|
"TEL;VALUE=uri;TYPE=\"work,cell,voice,video,text\":tel:+1-418-262-6501\r\n" &
|
||||||
|
"EMAIL;TYPE=work:simon.perreault@viagenie.ca\r\n" &
|
||||||
|
"GEO;TYPE=work:geo:46.772673,-71.282945\r\n" &
|
||||||
|
"KEY;TYPE=work;VALUE=uri:\r\n" &
|
||||||
|
" http://www.viagenie.ca/simon.perreault/simon.asc\r\n" &
|
||||||
|
"TZ:-0500\r\n" &
|
||||||
|
"URL;TYPE=home:http://nomis80.org\r\n" &
|
||||||
|
"END:VCARD\r\n"
|
||||||
|
|
||||||
|
let vcards = parseVCards(vcardStr)
|
||||||
|
check vcards.len == 1
|
||||||
|
let sp = cast[VCard4](vcards[0])
|
||||||
|
check:
|
||||||
|
sp.fn.len == 1
|
||||||
|
sp.fn[0].value == "Simon Perreault"
|
||||||
|
sp.gender.isSome
|
||||||
|
sp.gender.get.sex == some(VC4_Sex.Male)
|
||||||
|
sp.gender.get.genderIdentity.isNone
|
||||||
|
sp.lang.len == 2
|
||||||
|
sp.lang --> map(it.value) == @["fr", "en"]
|
||||||
|
|
||||||
|
test "custom properties are serialized":
|
||||||
|
let email = newVC4_Email(
|
||||||
|
value ="john.smith@testco.test",
|
||||||
|
types = @["work", "internet"],
|
||||||
|
params = @[("PREF", @["1"]), ("X-ATTACHMENT-LIMIT", @["25MB"])])
|
||||||
|
|
||||||
|
check serialize(email) ==
|
||||||
|
"EMAIL;X-ATTACHMENT-LIMIT=25MB;TYPE=work,internet;PREF=1:john.smith@testco.test"
|
||||||
|
|
||||||
|
test "can parse properties with escaped characters":
|
||||||
|
check v4Ex.note.len == 1
|
||||||
|
let note = v4Ex.note[0]
|
||||||
|
|
||||||
|
check note.value ==
|
||||||
|
"This is an example, for clarity; in text value cases the parser " &
|
||||||
|
"will recognize escape values for ',', '\\', and newlines. For " &
|
||||||
|
"example:" &
|
||||||
|
"\n\t123 Flagstaff Road" &
|
||||||
|
"\n\tPlaceville, MA"
|
||||||
|
|
||||||
|
test "can parse parameters with escaped characters":
|
||||||
|
let prop = v4Ex.customProp("X-CUSTOM-EXAMPLE")[0]
|
||||||
|
check prop.value ==
|
||||||
|
"This is an example, for clarity; in straight value cases, the parser " &
|
||||||
|
"does not recognize any escape values, as the meaning of the content " &
|
||||||
|
"is implementation-specific."
|
||||||
|
let param1 = prop.params --> filter(it.name == "PARAM")
|
||||||
|
let label = prop.params --> filter(it.name == "LABEL")
|
||||||
|
check:
|
||||||
|
param1.len == 1
|
||||||
|
param1[0].values == @["How one says, \"Hello.\""]
|
||||||
|
label.len == 1
|
||||||
|
label[0].values == @["^top\nsecond line"]
|
||||||
|
|
||||||
|
test "Data URIs are parsed correctly":
|
||||||
|
let expectedB64 = readFile("tests/allen.foster.jpg.uri")
|
||||||
|
|
||||||
|
check:
|
||||||
|
v4Ex.photo.len == 2
|
||||||
|
v4Ex.photo[0].altId == some("1")
|
||||||
|
v4Ex.photo[0].value ==
|
||||||
|
"https://tile.loc.gov/storage-services/service/pnp/bellcm/02200/02297r.jpg"
|
||||||
|
v4Ex.photo[0].valueType == some("uri")
|
||||||
|
v4Ex.photo[1].altId == some("1")
|
||||||
|
v4Ex.photo[1].value == expectedB64
|
||||||
|
v4Ex.photo[1].valueType.isNone
|
||||||
|
|
||||||
|
test "URI-type properties are parsed correctly":
|
||||||
|
# Covers SOURCE, PHOTO, IMPP, GEO, LOGO, MEMBER, SOUND, URL, FBURL,
|
||||||
|
# CALADRURI, and CALURI
|
||||||
|
check:
|
||||||
|
v4Ex.source.len == 1
|
||||||
|
v4Ex.source[0].value == "https://carddav.fosters.test/allen.vcf"
|
||||||
|
v4Ex.source[0].valueType == some("uri")
|
||||||
|
v4Ex.url.len == 1
|
||||||
|
v4Ex.url[0].value == "https://allen.fosters.test/"
|
||||||
|
|
||||||
|
test "URI-type properties are serialized correctly":
|
||||||
|
# Covers SOURCE, PHOTO, IMPP, GEO, LOGO, MEMBER, SOUND, URL, FBURL,
|
||||||
|
# CALADRURI, and CALURI
|
||||||
|
let src = newVC4_Source(value="https://carddav.example.test/john-smith.vcf")
|
||||||
|
check serialize(src) == "SOURCE:https://carddav.example.test/john-smith.vcf"
|
||||||
|
|
||||||
|
test "Single-text properties are parsed correctly":
|
||||||
|
# Covers KIND, XML, FN, NICKNAME, EMAIL, LANG, TZ, TITLE, ROLE, ORG, NOTE,
|
||||||
|
# PRODID, and VERSION
|
||||||
|
check:
|
||||||
|
v4Ex.kind.isSome
|
||||||
|
v4Ex.kind.get.value == "individual"
|
||||||
|
v4Ex.nickname.len == 2
|
||||||
|
v4Ex.nickname[0].value == @["Jack Jr."]
|
||||||
|
v4Ex.nickname[1].value == @["Doc A"]
|
||||||
|
v4Ex.fn.len == 1
|
||||||
|
v4Ex.fn[0].value == "Dr. Allen Foster"
|
||||||
|
v4Ex.email.len == 2
|
||||||
|
v4Ex.email[0].value == "jack.foster@company.test"
|
||||||
|
v4Ex.email[0].types == @["work"]
|
||||||
|
|
||||||
|
test "URI or Text properties are parsed correctly":
|
||||||
|
# Covers TEL, RELATED, UID, KEY
|
||||||
|
check:
|
||||||
|
v4Ex.tel.len == 3
|
||||||
|
v4ex.tel[0].types == @[$VC4_TelType.ttCell]
|
||||||
|
v4Ex.tel[0].value == "+1 555-123-4567"
|
||||||
|
v4Ex.tel[2].types == @[$VC4_TelType.ttWork,$VC4_TelType.ttVoice]
|
||||||
|
v4Ex.tel[2].valueType == some($vtUri)
|
||||||
|
v4Ex.tel[2].value == "tel:+1-555-874-1234"
|
||||||
|
|
||||||
|
test "N is parsed correctly":
|
||||||
|
check:
|
||||||
|
v4Ex.n.isSome
|
||||||
|
v4Ex.n.get.given == @["Jack"]
|
||||||
|
v4Ex.n.get.family == @["Foster"]
|
||||||
|
v4Ex.n.get.additional == @["John", "Allen"]
|
||||||
|
v4Ex.n.get.prefixes == @["Dr."]
|
||||||
|
v4Ex.n.get.suffixes == @["II"]
|
||||||
|
|
||||||
|
test "BDAY is parsed correctly":
|
||||||
|
check:
|
||||||
|
v4Ex.bday.isSome
|
||||||
|
v4Ex.bday.get.value == "--1224"
|
||||||
|
v4Ex.bday.get.year.isNone
|
||||||
|
v4Ex.bday.get.month == some(12)
|
||||||
|
v4Ex.bday.get.day == some(24)
|
||||||
|
|
||||||
|
test "ANNIVERSARY is parsed correctly":
|
||||||
|
check:
|
||||||
|
v4Ex.anniversary.isSome
|
||||||
|
v4Ex.anniversary.get.value == "20140612T163000-0500"
|
||||||
|
v4Ex.anniversary.get.year == some(2014)
|
||||||
|
v4Ex.anniversary.get.hour == some(16)
|
||||||
|
v4Ex.anniversary.get.minute == some(30)
|
||||||
|
v4Ex.anniversary.get.timezone == some("-0500")
|
||||||
|
|
||||||
|
test "GENDER is parsed correctly":
|
||||||
|
check:
|
||||||
|
v4Ex.gender.isSome
|
||||||
|
v4Ex.gender.get.sex == some(VC4_Sex.Male)
|
||||||
|
v4Ex.gender.get.genderIdentity == some("male")
|
||||||
|
|
||||||
|
#[
|
||||||
|
test "CATEGORIES is parsed correctly":
|
||||||
|
test "REV is parsed correctly":
|
||||||
|
test "CLIENTPIDMAP is parsed correctly":
|
||||||
|
]#
|
||||||
|
|
||||||
|
test "unknown properties are parsed correctly":
|
||||||
|
|
||||||
|
check v4Ex.customProp("MADE-UP-PROP").len == 1
|
||||||
|
let madeUpProp = v4Ex.customProp("MADE-UP-PROP")[0]
|
||||||
|
check:
|
||||||
|
madeUpProp.name == "MADE-UP-PROP"
|
||||||
|
madeUpProp.value == "Sample value for my made-up prop."
|
||||||
|
|
||||||
|
let cardWithAltBdayStr = testVCardTemplate % [(
|
||||||
|
"BDAY;VALUE=text;ALTID=1:20th century\r\n" &
|
||||||
|
"BDAY;VALUE=date-and-or-time;ALTID=1:19650321\r\n"
|
||||||
|
)]
|
||||||
|
|
||||||
|
test "single-cardinality properties allow multiples with ALTID":
|
||||||
|
check parseVCards(cardWithAltBdayStr).len == 1
|
||||||
|
|
||||||
|
let hasAltBdays = cast[VCard4](parseVCards(cardWithAltBdayStr)[0])
|
||||||
|
|
||||||
|
test "properties with cardinality 1 and altids return the first found by default":
|
||||||
|
check:
|
||||||
|
hasAltBdays.bday.isSome
|
||||||
|
hasAltBdays.bday.get.value == "20th century"
|
||||||
|
hasAltBdays.bday.get.year.isNone
|
||||||
|
|
||||||
|
test "allAlternatives":
|
||||||
|
check:
|
||||||
|
hasAltBdays.content.len == 3
|
||||||
|
hasAltBdays.bday.isSome
|
||||||
|
|
||||||
|
let allBdays = allAlternatives[VC4_Bday](hasAltBdays)
|
||||||
|
check:
|
||||||
|
allBdays.len == 1
|
||||||
|
allBdays.contains("1")
|
||||||
|
allBdays["1"].len == 2
|
||||||
|
|
||||||
|
let bday0 = allBdays["1"][0]
|
||||||
|
check:
|
||||||
|
bday0.value == "20th century"
|
||||||
|
bday0.year.isNone
|
||||||
|
bday0.month.isNone
|
||||||
|
bday0.day.isNone
|
||||||
|
bday0.hour.isNone
|
||||||
|
bday0.minute.isNone
|
||||||
|
bday0.second.isNone
|
||||||
|
bday0.timezone.isNone
|
||||||
|
|
||||||
|
let bday1 = allBDays["1"][1]
|
||||||
|
check:
|
||||||
|
bday1.value == "19650321"
|
||||||
|
bday1.year == some(1965)
|
||||||
|
bday1.month == some(3)
|
||||||
|
bday1.day == some(21)
|
||||||
|
bday1.hour.isNone
|
||||||
|
bday1.minute.isNone
|
||||||
|
bday1.second.isNone
|
||||||
|
|
||||||
|
test "PREF ordering":
|
||||||
|
check:
|
||||||
|
v4Ex.nickname --> map(it.value) == @[@["Jack Jr."], @["Doc A"]]
|
||||||
|
v4Ex.nickname.inPrefOrder --> map(it.value) == @[@["Doc A"], @["Jack Jr."]]
|
@ -1,6 +1,6 @@
|
|||||||
# Package
|
# Package
|
||||||
|
|
||||||
version = "0.1.1"
|
version = "0.2.0"
|
||||||
author = "Jonathan Bernard"
|
author = "Jonathan Bernard"
|
||||||
description = "Nim parser for the vCard format version 3.0 (4.0 planned)."
|
description = "Nim parser for the vCard format version 3.0 (4.0 planned)."
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user