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:
parent
0ec1856d1b
commit
935f1bae2f
1
.gitignore
vendored
1
.gitignore
vendored
@ -3,4 +3,5 @@ tests/*
|
||||
|
||||
|
||||
bin/
|
||||
doc/
|
||||
*.sw?
|
||||
|
11
Makefile
11
Makefile
@ -1,13 +1,18 @@
|
||||
# Make does not offer a recursive wildcard function, so here's one:
|
||||
rwildcard=$(wildcard $1$2) $(foreach d,$(wildcard $1*),$(call rwildcard,$d/,$2))
|
||||
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
|
||||
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:
|
||||
|
@ -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
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))
|
@ -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)
|
@ -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
|
||||
|
@ -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)):
|
||||
|
@ -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
|
||||
|
Loading…
x
Reference in New Issue
Block a user