WIP documentation

- The documentation is cluttered enough as it is with the large number
  of procedures supporting vCard 3 and 4. Split common out into the
  publicly exposed bits and the private internals. This makes it obvious
  which common functionality a client can expect to have exposed on the
  main vcard module.

- Add documentation (WIP) on the vcard3 module.
This commit is contained in:
Jonathan Bernard 2023-05-03 02:16:18 -05:00
parent 0ec1856d1b
commit 935f1bae2f
8 changed files with 401 additions and 318 deletions

1
.gitignore vendored
View File

@ -3,4 +3,5 @@ tests/*
bin/
doc/
*.sw?

View File

@ -6,8 +6,13 @@ TEST_SOURCES=$(wildcard tests/*.nim)
TESTS=$(patsubst %.nim,bin/%,$(TEST_SOURCES))
.PHONY: build
build: test
nimble 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:

View File

@ -2,26 +2,28 @@
# © 2022 Jonathan Bernard
## The `vcard` module implements a high-performance vCard parser for both
## versions 3.0 (defined by RFCs [2425][rfc2425] and [2426][rfc2426]) and 4.0
## (defined by RFC [6350][rfc6350])
## versions 3.0 (defined by RFCs 2425_ and 2426_) and 4.0 (defined by RFC
## 6350_)
##
## [rfc2425]: https://tools.ietf.org/html/rfc2425
## [rfc2426]: https://tools.ietf.org/html/rfc2426
## [rfc6350]: https://tools.ietf.org/html/rfc6350
## .. _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/[common, lexer]
import ./vcard/[vcard3, vcard4]
import ./vcard/private/[internals, lexer]
import ./vcard/[common, vcard3, vcard4]
export vcard3, vcard4
export common.VC_XParam,
export common.VC_Param,
common.VC_XParam,
common.VCard,
common.VCardParsingError,
common.VCardVersion,
common.VCard,
common.getSingleValue,
common.getMultipleValues
common.allPropsOfType,
common.getMultipleValues,
common.getSingleValue
proc add[T](vc: VCard, content: varargs[T]): void =
proc add*[T](vc: VCard, content: varargs[T]): void =
if vc.parsedVersion == VCardV3: add(cast[VCard3](vc), content)
else: add(cast[VCard4](vc), content)
@ -53,12 +55,18 @@ proc readVCard*(p: var VCardParser): VCard =
if result.parsedVersion == VCardV3:
while (p.skip(CRLF, true)): discard
proc parseVCards*(input: Stream, filename = "input"): seq[VCard] =
var p: VCardParser
p.filename = filename
lexer.open(p, input)
proc initVCardParser*(input: Stream, filename = "input"): VCardParser =
result.filename = filename
lexer.open(result, input)
# until EOF
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] =

91
src/vcard/common.nim Normal file
View 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))

View File

@ -6,56 +6,27 @@
## implementations.
##
## This module is not intended to be exposed to library consumers. It is
## intended to be imported by the vcard3 and vcard4 implementations. There are
## a handful of functions (under the **Public Definitions** section) that will
## be re-exported by the root `vcard` module and available on that namespace to
## users of the library.
## 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
## Internal Types (used by `vcard{3,4}`)
## =====================================
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.
## enum used to define the possible cardinalities of vCard properties.
vpcAtMostOne,
vpcExactlyOne,
vpcAtLeastOne
vpcAny
## External Types (exported on `vcard`)
## ====================================
type
VC_Param* = tuple[name: string, values: seq[string]]
## Representation of VCard parameter and its values.
VCardVersion* = enum VCardV3 = "3.0", VCardV4 = "4.0" ## \
## enum used to differentiate VCard3 and VCard4 versions.
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
## Internal constants (used by `vcard{3,4}`)
## =========================================
# Internal constants (used by `vcard{3,4}`)
# =========================================
const CRLF* = "\r\n"
const WSP* = {' ', '\t'}
const DIGIT* = { '0'..'9' }
@ -79,8 +50,8 @@ const DATE_TIME_FMTS = [
const ALL_DATE_AND_OR_TIME_FMTS = DATE_TIME_FMTS.toSeq & DATE_FMTS.toSeq
## Internal Utility/Implementation Functions
## =========================================
# Internal Utility/Implementation Functions
# =========================================
proc parseDateTimeStr(
dateStr: string,
@ -127,10 +98,8 @@ macro assignFields*(assign: untyped, fields: varargs[untyped]): untyped =
exp.add(f, f)
result.add(exp)
## Internal Parsing Functionality
## ==============================
proc getSingleValue*(params: openarray[VC_Param], name: string): Option[string];
# Internal Parsing Functionality
# ==============================
proc error*(p: VCardParser, msg: string) =
raise newException(VCardParsingError, "$1($2, $3) Error: $4" %
@ -187,7 +156,7 @@ proc isNext*(p: var VCardParser, expected: string, caseSensitive = false): bool
p.returnToBookmark
proc readGroup*(p: var VCardParser): Option[string] =
## All VCARD content items can be optionally prefixed with a group name. This
## 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
@ -302,8 +271,8 @@ proc validateRequiredParameters*(
if pv.isSome and pv.get != v:
p.error("parameter '$1' must have the value '$2'" % [n, v])
## Internal Serialization Utilities
## ================================
# Internal Serialization Utilities
# ================================
func foldContentLine*(s: string): string =
result = ""
@ -312,61 +281,3 @@ func foldContentLine*(s: string): string =
result &= rem[0..<75] & "\r\n "
rem = rem[75..^1]
result &= rem
## Publicly Exported Procedure and Functions
## =========================================
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 VCard3 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] = findAll[T](vc)
## 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:
##
## let vc3 = parseVCards(...)
## let allNames = allPropsOfType[VC3_Name](vc3)

View File

@ -1,9 +1,9 @@
# VCard-specific Lexer
# vCard-specific Lexer
# © 2022-2023 Jonathan Bernard
## This module defines a lexer with functionality useful for parsing VCard
## This module defines a lexer with functionality useful for parsing vCard
## content. Specifically:
## - it understands the VCard line-folding logic and transparently joins folded
## - 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

View File

@ -1,21 +1,22 @@
# vCard 3.0 and 4.0 Nim implementation
# vCard 3.0 implementation
# © 2022 Jonathan Bernard
## The `vcard` module implements a high-performance vCard parser for both
## versions 3.0 (defined by RFCs [2425][rfc2425] and [2426][rfc2426]) and 4.0
## (defined by RFC [6350][rfc6350])
## This module implements a high-performance vCard parser for vCard version
## 3.0 (defined in RFCs 2425_ and 2426_).
##
## [rfc2425]: https://tools.ietf.org/html/rfc2425
## [rfc2426]: https://tools.ietf.org/html/rfc2426
## [rfc6350]: https://tools.ietf.org/html/rfc6350
## .. _rfc2425: https://tools.ietf.org/html/rfc2425
## .. _rfc2426: https://tools.ietf.org/html/rfc2426
import std/[base64, genasts, macros, options, sequtils, streams, strutils,
tables, times, unicode]
import zero_functional
import ./private/[common, lexer]
import ./common
import ./private/[internals, lexer]
# Internal enumerations used to capture the value types and properties defined
# by the spec and constants used.
type
VC3_ValueTypes = enum
vtUri = "uri",
@ -65,177 +66,6 @@ type
pnClass = "CLASS"
pnKey = "KEY"
VC3_Property* = ref object of RootObj
propertyId: int
group*: Option[string]
name*: string
VC3_SimpleTextProperty* = ref object of VC3_Property
value*: string
isPText*: bool # true if VALUE=ptext, false by default
language*: Option[string]
xParams: seq[VC_XParam]
VC3_BinaryProperty* = ref object of VC3_Property
valueType*: Option[string] # binary / uri. Stored separately from ENCODING
# (captured in the isInline field) because the
# VALUE parameter is not set by default, but is
# allowed to be set.
value*: string # either a URI or bit sequence, both stored as string
binaryType*: Option[string]# if using ENCODING=b, there may also be a TYPE
# parameter specifying the MIME type of the
# binary-encoded object, which is stored here.
isInline*: bool # true if ENCODING=b, false by default
VC3_Name* = ref object of VC3_Property
value*: string
VC3_Profile* = ref object of VC3_Property
VC3_Source* = ref object of VC3_Property
valueType*: Option[string] # uri
value*: string # URI
context*: Option[string]
xParams*: seq[VC_XParam]
VC3_Fn* = ref object of VC3_SimpleTextProperty
VC3_N* = ref object of VC3_Property
family*: seq[string]
given*: seq[string]
additional*: seq[string]
prefixes*: seq[string]
suffixes*: seq[string]
language*: Option[string]
isPText*: bool # true if VALUE=ptext, false by default
xParams*: seq[VC_XParam]
VC3_Nickname* = ref object of VC3_SimpleTextProperty
VC3_Photo* = ref object of VC3_BinaryProperty
VC3_Bday* = ref object of VC3_Property
valueType*: Option[string] # date / date-time
value*: DateTime
VC3_AdrTypes* = enum
# Standard types defined in RFC2426
atDom = "DOM"
atIntl = "INTL"
atPostal = "POSTAL"
atParcel = "PARCEL"
atHome = "HOME"
atWork = "WORK"
atPref = "PREF"
VC3_Adr* = ref object of VC3_Property
adrType*: seq[string]
poBox*: string
extendedAdr*: string
streetAdr*: string
locality*: string
region*: string
postalCode*: string
country*: string
isPText*: bool # true if VALUE=ptext, false by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Label* = ref object of VC3_SimpleTextProperty
adrType*: seq[string]
VC3_TelTypes* = enum
ttHome = "HOME",
ttWork = "WORK",
ttPref = "PREF",
ttVoice = "VOICE",
ttFax = "FAX",
ttMsg = "MSG",
ttCell = "CELL",
ttPager = "PAGER",
ttBbs = "BBS",
ttModem = "MODEM",
ttCar = "CAR",
ttIsdn = "ISDN",
ttVideo = "VIDEO",
ttPcs = "PCS"
VC3_Tel* = ref object of VC3_Property
telType*: seq[string]
value*: string
VC3_EmailType* = enum
etInternet = "INTERNET",
etX400 = "X400"
VC3_Email* = ref object of VC3_Property
emailType*: seq[string]
value*: string
VC3_Mailer* = ref object of VC3_SimpleTextProperty
VC3_TZ* = ref object of VC3_Property
value*: string
isText*: bool # true if VALUE=text, false by default
VC3_Geo* = ref object of VC3_Property
lat*, long*: float
VC3_Title* = ref object of VC3_SimpleTextProperty
VC3_Role* = ref object of VC3_SimpleTextProperty
VC3_Logo* = ref object of VC3_BinaryProperty
VC3_Agent* = ref object of VC3_Property
value*: string # either an escaped vCard object, or a URI
isInline*: bool # false if VALUE=uri, true by default
VC3_Org* = ref object of VC3_Property
value*: seq[string]
isPText*: bool # true if VALUE=ptext, false by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Categories* = ref object of VC3_Property
value*: seq[string]
isPText*: bool # true if VALUE=ptext, false by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Note* = ref object of VC3_SimpleTextProperty
VC3_Prodid* = ref object of VC3_SimpleTextProperty
VC3_Rev* = ref object of VC3_Property
valueType*: Option[string] # date / date-time
value*: DateTime
VC3_SortString* = ref object of VC3_SimpleTextProperty
VC3_Sound* = ref object of VC3_BinaryProperty
VC3_UID* = ref object of VC3_Property
value*: string
VC3_URL* = ref object of VC3_Property
value*: string
VC3_Version* = ref object of VC3_Property
value*: string # 3.0
VC3_Class* = ref object of VC3_Property
value*: string
VC3_Key* = ref object of VC3_BinaryProperty
keyType*: Option[string] # x509 / pgp
VC3_XType* = ref object of VC3_SimpleTextProperty
VCard3* = ref object of VCard
nextPropertyId: int
content*: seq[VC3_Property]
const propertyCardMap: Table[VC3_PropertyName, VC_PropCardinality] = [
(pnName, vpcAtMostOne),
(pnProfile, vpcAtMostOne),
@ -273,8 +103,227 @@ const propertyCardMap: Table[VC3_PropertyName, VC_PropCardinality] = [
const DATE_FMT = "yyyy-MM-dd"
const DATETIME_FMT = "yyyy-MM-dd'T'HH:mm:sszz"
## Externally Exposed Types
## ========================
##
## The following are the object types for the vCard 3.0 implementation
## interface.
type
VC3_Property* = ref object of RootObj
## Abstract base class for all vCard 3.0 property objects.
propertyId: int
group*: Option[string]
name*: string
VC3_SimpleTextProperty* = ref object of VC3_Property
## Abstract base class for all vCard 3.0 properties that only support the
## text value type (`VALUE=text` or `VALUE=ptext`).
value*: string
isPText*: bool ## true if VALUE=ptext, false by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_BinaryProperty* = ref object of VC3_Property
## Abstract base class for all vCard 3.0 properties that support the binary
## or uri value types (`ENCODING=b` or `VALUE=uri`).
valueType*: Option[string] ## \
## binary / uri. Stored separately from ENCODING (captured in the
## isInline field) because the VALUE parameter is not set by default, but
## is allowed to be set.
value*: string ## \
## either a URI or bit sequence, both stored as string
binaryType*: Option[string] ## \
## if using ENCODING=b, there may also be a TYPE parameter specifying the
## MIME type of the binary-encoded object, which is stored here.
isInline*: bool ## \
## true if ENCODING=b, false by default
VC3_Name* = ref object of VC3_Property
## Concrete class representing vCard 3.0 NAME properties.
value*: string
VC3_Profile* = ref object of VC3_Property
## Concrete class representing vCard 3.0 PROFILE properties.
VC3_Source* = ref object of VC3_Property
## Concrete class representing vCard 3.0 SOURCE properties.
valueType*: Option[string] ## If set, this must be set to "uri"
value*: string ## The URI value.
context*: Option[string]
xParams*: seq[VC_XParam]
VC3_Fn* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 FN properties.
VC3_N* = ref object of VC3_Property
## Concrete class representing vCard 3.0 N properties.
family*: seq[string] ## Surname / family name
given*: seq[string] ## First name / given name
additional*: seq[string] ## Additional / middle names
prefixes*: seq[string] ## e.g. Mr., Dr., etc.
suffixes*: seq[string] ## e.g. Esq., II, etc.
language*: Option[string]
isPText*: bool ## true if VALUE=ptext, false by default
xParams*: seq[VC_XParam]
VC3_Nickname* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 NICKNAME properties.
VC3_Photo* = ref object of VC3_BinaryProperty
## Concrete class representing vCard 3.0 PHOTO properties.
VC3_Bday* = ref object of VC3_Property
## Concrete class representing vCard 3.0 BDAY properties.
valueType*: Option[string] # date / date-time
value*: DateTime
VC3_AdrTypes* = enum
## Standard types for ADR values defined in RFC2426
atDom = "DOM"
atIntl = "INTL"
atPostal = "POSTAL"
atParcel = "PARCEL"
atHome = "HOME"
atWork = "WORK"
atPref = "PREF"
VC3_Adr* = ref object of VC3_Property
## Concrete class representing vCard 3.0 ADR properties.
adrType*: seq[string]
poBox*: string
extendedAdr*: string
streetAdr*: string
locality*: string
region*: string
postalCode*: string
country*: string
isPText*: bool ## `true` if `VALUE=ptext`, `false` by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Label* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 LABEL properties.
adrType*: seq[string]
VC3_TelTypes* = enum
## Standard types for TEL values defined in RFC2426
ttHome = "HOME",
ttWork = "WORK",
ttPref = "PREF",
ttVoice = "VOICE",
ttFax = "FAX",
ttMsg = "MSG",
ttCell = "CELL",
ttPager = "PAGER",
ttBbs = "BBS",
ttModem = "MODEM",
ttCar = "CAR",
ttIsdn = "ISDN",
ttVideo = "VIDEO",
ttPcs = "PCS"
VC3_Tel* = ref object of VC3_Property
## Concrete class representing vCard 3.0 TEL properties.
telType*: seq[string]
value*: string
VC3_EmailType* = enum
## Standard types for EMAIL values defined in RFC2426
etInternet = "INTERNET",
etX400 = "X400"
VC3_Email* = ref object of VC3_Property
## Concrete class representing vCard 3.0 EMAIL properties.
emailType*: seq[string]
value*: string
VC3_Mailer* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 MAILER properties.
VC3_TZ* = ref object of VC3_Property
## Concrete class representing vCard 3.0 TZ properties.
value*: string
isText*: bool ## `true` if `VALUE=text`, `false` by default
VC3_Geo* = ref object of VC3_Property
## Concrete class representing vCard 3.0 GEO properties.
lat*, long*: float
VC3_Title* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 TITLE properties.
VC3_Role* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 ROLE properties.
VC3_Logo* = ref object of VC3_BinaryProperty
## Concrete class representing vCard 3.0 LOGO properties.
VC3_Agent* = ref object of VC3_Property
## Concrete class representing vCard 3.0 AGENT properties.
value*: string ## either an escaped vCard object, or a URI
isInline*: bool ## `false` if `VALUE=uri`, `true` by default
VC3_Org* = ref object of VC3_Property
## Concrete class representing vCard 3.0 ORG properties.
value*: seq[string]
isPText*: bool ## `true` if `VALUE=ptext`, `false` by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Categories* = ref object of VC3_Property
## Concrete class representing vCard 3.0 CATEGORIES properties.
value*: seq[string]
isPText*: bool ## `true` if `VALUE=ptext`, `false` by default
language*: Option[string]
xParams*: seq[VC_XParam]
VC3_Note* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 NOTE properties.
VC3_Prodid* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 PRODID properties.
VC3_Rev* = ref object of VC3_Property
## Concrete class representing vCard 3.0 REV properties.
valueType*: Option[string] # date / date-time
value*: DateTime
VC3_SortString* = ref object of VC3_SimpleTextProperty
## Concrete class representing vCard 3.0 SORT-STRING properties.
VC3_Sound* = ref object of VC3_BinaryProperty
## Concrete class representing vCard 3.0 SOUND properties.
VC3_UID* = ref object of VC3_Property
## Concrete class representing vCard 3.0 UID properties.
value*: string
VC3_URL* = ref object of VC3_Property
## Concrete class representing vCard 3.0 URL properties.
value*: string
VC3_Version* = ref object of VC3_Property
## Concrete class representing vCard 3.0 VERSION properties.
value*: string # 3.0
VC3_Class* = ref object of VC3_Property
## Concrete class representing vCard 3.0 CLASS properties.
value*: string
VC3_Key* = ref object of VC3_BinaryProperty
## Concrete class representing vCard 3.0 KEY properties.
keyType*: Option[string] # x509 / pgp
VC3_XType* = ref object of VC3_SimpleTextProperty
VCard3* = ref object of VCard
## Concrete class implementing the vCard 3.0 type.
nextPropertyId: int
content*: seq[VC3_Property]
# Internal Utility/Implementation
# =============================================================================
# ===============================
template takePropertyId(vc3: VCard3): int =
vc3.nextPropertyId += 1
@ -291,8 +340,8 @@ func namesForProp(prop: VC3_PropertyName):
ident("newVC3_" & name),
ident(name.toLower))
# Initializers
# =============================================================================
## Initializers
## ============
func newVC3_Name*(value: string, group = none[string]()): VC3_Name =
return VC3_Name(name: "NAME", value: value, group: group)
@ -585,14 +634,17 @@ func newVC3_XType*(
value, language, isPText, xParams, group)
# Accessors
# =============================================================================
# =========
func forGroup*(vc: openarray[VC3_Property], group: string): seq[VC3_Property] =
return vc.filterIt(it.group.isSome and it.group.get == group)
func forGroup*(vc3: VCard3, group: string): seq[VC3_Property] =
## Return all properties defined on the vCard associated with the given
## group.
return vc3.content.filterIt(it.group.isSome and it.group.get == group)
func groups*(vc: openarray[VC3_Property]): seq[string] =
func groups*(vc3: VCard3): seq[string] =
## Return a list of all groups present on the vCard
result = @[]
for c in vc:
for c in vc3.content:
if c.group.isSome:
let grp = c.group.get
if not result.contains(grp): result.add(grp)
@ -610,8 +662,11 @@ macro genPropertyAccessors(
let funcDef = genAstOpt({kDirtyTemplate}, funcName, typeName):
func funcName*(vc3: VCard3): Option[typeName] =
result = findFirst[typeName](vc3.content)
funcDef[6].insert(0, newCommentStmtNode(
"Return the single " & $pn & " property (if present)."))
result.add(funcDef)
of vpcExactlyOne:
let funcDef = genAstOpt({kDirtyTemplate}, funcName, pn, typeName):
func funcName*(vc3: VCard3): typeName =
@ -621,18 +676,23 @@ macro genPropertyAccessors(
"VCard should have exactly one $# property, but $# were found" %
[$pn, $props.len])
result = props[0]
funcDef[6].insert(0, newCommentStmtNode(
"Return the " & $pn & " property."))
result.add(funcDef)
of vpcAtLeastOne, vpcAny:
let funcDef = genAstOpt({kDirtyTemplate}, funcName, typeName):
func funcName*(vc3: VCard3): seq[typeName] =
result = findAll[typeName](vc3.content)
funcDef[6].insert(0, newCommentStmtNode(
"Return all instances of the " & $pn & " property."))
result.add(funcDef)
genPropertyAccessors(propertyCardMap.pairs.toSeq -->
filter(not [pnVersion].contains(it[0])))
func version*(vc3: VCard3): VC3_Version =
## Return the VERSION property.
let found = findFirst[VC3_Version](vc3.content)
if found.isSome: return found.get
else: return VC3_Version(
@ -641,13 +701,17 @@ func version*(vc3: VCard3): VC3_Version =
name: "VERSION",
value: "3.0")
func xTypes*(c: openarray[VC3_Property]): seq[VC3_XType] = findAll[VC3_XType](c)
func xTypes*(vc3: VCard3): seq[VC3_XType] = vc3.content.xTypes
func xTypes*(vc3: VCard3): seq[VC3_XType] = findAll[VC3_XType](vc3.content)
## Return all extended properties (starting with `x-`).
# Setters
# =============================================================================
func set*[T: VC3_Property](vc3: VCard3, content: varargs[T]): void =
## Set the given property on the vCard. This will replace the first existing
## instance of this property or append this as a new property if there is no
## existing instance of this property. This is useful for properties which
## should only appear once within the vCard.
for c in content:
var nc = c
let existingIdx = vc3.content.indexOfIt(it of T)
@ -659,6 +723,7 @@ func set*[T: VC3_Property](vc3: VCard3, content: varargs[T]): void =
vc3.content[existingIdx] = nc
func add*[T: VC3_Property](vc3: VCard3, content: varargs[T]): void =
## Add a new property to the end of the vCard.
for c in content:
var nc = c
nc.propertyId = vc3.takePropertyId
@ -811,6 +876,7 @@ proc serialize(c: VC3_Property): string =
return serialize(cast[VC3_BinaryProperty](c))
proc `$`*(vc3: VCard3): string =
## Serialize a vCard into its textual representation.
result = "BEGIN:VCARD" & CRLF
result &= "VERSION:3.0" & CRLF
for c in vc3.content.filterIt(not (it of VC3_Version)):

View File

@ -4,7 +4,8 @@ import zero_functional
#from std/sequtils import toSeq
import ./private/[common, lexer]
import ./common
import ./private/[internals, lexer]
type
VC4_ValueType* = enum