- Allow
.letOrNull((pick) => null)
to returnnull
without manually setting a nullable type #61 - CI 2024 update
- Remove long deprecated methods:
addContext
,asBool
,asDateTime
,asDouble
,asInt
,asList
,asMap
,asString
. Use theiras*OrThrow
replacements. - Add support for timezones in
asDateTime*
methods #47, #51 - Add official support for date formats
RFC 3339
,RFC 2822
andRFC 1036
- Push test coverage to 100% 🤘
- New: Support for more date formats.
asDateTime*
received an optionalformat
parameter. By default, all possible formats will be parsed. To the existingISO 8601
format,RFC 1123
,RFC 850
andasctime
have been added which are typically used for the HTTP header or cookies. - Documentation added for
asDouble
andasMap
-
New:
pickFromJson(json, args...)
allows parsing of a json String, without manually callingjsonDecode
#41 -
New:
pickDeep(json, 'some.key.inside.the.object'.split('.'))
allows picking with a dynamic depth #40 -
Add
Pick.index
to get the element index for list items #38pick(["John", "Paul", "George", "Ringo"]).asListOrThrow((pick) { final index = pick.index!; return Artist(id: index, name: pick.asStringOrThrow()); );
-
Pick.asIntOrThrow()
now allows parsing of doubles when one of the newroundDouble
ortruncateDouble
parameters istrue
#37. Thx @stevendz -
Add dartdoc to
asList*()
extensions
- Deprecated parsing extensions of
RequiredPick
to acknowledge that all parsers eventually causes errors. From now on, always use.asIntOrThrow()
instead of.required().asInt()
. Only exception is.required().toString()
. Read more in #34 - Replace
dynamic
withObject
where possible - Rename
Pick.location()
toPick.debugParsingExit
- Removal of
PickLocaiton
andPickContext
mixins. They are now part ofPick
RequiredPick
now extendsPick
making it easier to write parsers for custom types
- Enable nullsafety (requires Dart >=2.12)
Backports 0.8.0 to pre-nullsafety
-
Remove long deprecated
parseJsonTo*
methods. Use thepick(json, args*)
api -
New
asXyzOrThrow()
methods as shorthand for.required().asXyz()
featuring better error messagesasBoolOrThrow()
asDateTimeOrThrow()
asDoubleOrThrow()
asIntOrThrow()
letOrThrow()
asListOrThrow()
asMapOrThrow()
asStringOrThrow()
-
New
Pick.isAbsent
getter to check if a value is absent ornull
#24. Absent could mean- Accessing a key which doesn't exist in a
Map
- Reading the value from
List
when the index is greater than the length - Trying to access a key in a
Map
but the found data aObject
which isn't a Map
- Accessing a key which doesn't exist in a
-
New
RequiredPick.nullable()
converting aRequiredPick
back to aPick
with potentialnull
value -
New
PickLocation.followablePath
. WhilePickLocation.path
contains the full path to the value,followablePath
contains only the part which could be followed with a non-nullable value -
Breaking
asList*()
now requires the mapping function. -
Breaking
asList*()
now ignoresnull
values when parsing. The map function now receives aRequiredPick
as fist parameter instead of aPick
with a potential null value, making parsing easier.Therefore
pick().required().asList((RequiredPick pick) => /*...*/)
only maps non-nullable values. When your lists containnull
it will be ignored. This is fine in most cases and simplifies the map function.In rare cases, where your lists contain
null
values with meaning, use the second parameterwhenNull
to map those null values.asList((pick) => Person.fromPick(pick), whenNull: (Pick it) => null)
. The function still receives aPick
which gives access to thecontext
api or thePickLocation
. But thePick
never holds any value.
-
The String
"true"
and"false"
are now parsed as boolean -
Breaking Don't parse doubles as int because the is no rounding method which satisfies all #31
-
Breaking Allow parsing of "german" doubles with
,
as decimal separator #30 -
Improve error messages with more details where parsing stopped
-
Breaking
asList*()
methods now ignorenull
values. The map function now receives aRequiredPick
as fist parameter instead of aPick
making parsing easier.Therefore
pick().required().asList((RequiredPick pick) => /*...*/)
only maps non-nullable values. When your lists containnull
it will be ignored. This is fine in most cases and simplifies the map function.In rare cases, where your lists contain
null
values with meaning, use the second parameterwhenNull
to map those null values.asList((pick) => Person.fromPick(pick), whenNull: (Pick it) => null)
. The function still receives aPick
which gives access to thecontext
api or thePickLocation
. But thePick
never holds any value. -
Breaking Don't parse doubles as int because the is no rounding method which satisfies all #31
-
Breaking Allow parsing of "german" doubles with
,
as decimal separator #30 -
Improve error messages with more details where parsing stopped
-
New
RequiredPick.nullable()
converting aRequiredPick
back to aPick
with potentialnull
value -
New
PickLocation.followablePath
. WhilePickLocation.path
contains the full path to the value,followablePath
contains only the part which could be followed with a non-nullable value
- New
asXyzOrThrow()
methods as shorthand for.required().asXyz()
featuring better error messagesasBoolOrThrow()
asDateTimeOrThrow()
asDoubleOrThrow()
asIntOrThrow()
letOrThrow()
asListOrThrow()
asMapOrThrow()
asStringOrThrow()
- New
Pick.isAbsent
getter to check if a value is absent ornull
#24. Absent could mean- Accessing a key which doesn't exist in a
Map
- Reading the value from
List
when the index is greater than the length - Trying to access a key in a
Map
but the found data aObject
which isn't a Map
- Accessing a key which doesn't exist in a
- The String
"true"
and"false"
are now parsed as boolean - More nnbd refactoring
- Migrate to nullsafety (required Dart >=2.12)
- Remove long deprecated
parseJsonTo*
methods. Use thepick(json, args*)
api - Improve dartdoc
- Rename
Pick.addContext
toPick.withContext
using deprecation Pick.fromContext
now accepts 10 arguments for nested structures- Fix
Pick.fromContext
always returningcontext
not the value forkey
in contextMap
-
New context API. You can now attach relevant additional information for parsing directly to the
Pick
object. This allows passing information intofromPick
constructors without adding new parameters to all constructors in between.// Add context final shoes = pick(json, 'shoes') .addContext('apiVersion', "2.3.0") .addContext('lang', "en-US") .asListOrEmpty((p) => Shoe.fromPick(p.required()));
import 'package:version/version.dart'; // Read context factory Shoe.fromPick(RequiredPick pick) { // read context API final version = pick.fromContext('newApi').required().let((pick) => Version(pick.asString())); return Shoe( id: pick('id').required().asString(), name: pick('name').required().asString(), // manufacturer is a required field in the new API manufacturer: version >= Version(2, 3, 0) ? pick('manufacturer').required().asString() : pick('manufacturer').asStringOrNull(), tags: pick('tags').asListOrEmpty(), ); }
-
Breaking:
Pick
andRequiredPick
have chained their constructor signature.path
is now a named argument andcontext
has been added.
- RequiredPick(this.value, [this.path = const []])
+ RequiredPick(this.value, {this.path = const [], Map<String, dynamic> context})
- The
path
is now correctly forwarded afterPick#call
orPick#asListOrEmpty
and always shows the full path since origin
- Fix error reporting for
asMapOr[Empty|Null]
and don't swallow parsing errors - Throw Map cast errors when parsing, not lazily when accessing the data
- Fix error reporting of
asListOrNull(mapToUser)
andasListOrEmpty(mapToUser)
. Both now return errors during mapping and don't swallow them
- Print correct path in error message when json is
null
asDateTime()
now skips parsing when the value is already aDateTime
New APIs to map picks to objects and to map list elements to objects.
RequiredPick.let<R>(R Function(RequiredPick pick) block): R
Pick.letOrNull<R>(R Function(RequiredPick pick) block): R
RequiredPick.asList<T>([T Function(Pick) map]): List<T>
Pick.asListOrNull<T>([T Function(Pick) map]): List<T>
Pick.asListOrEmpty<T>([T Function(Pick) map]): List<T>
Here are two example how to actually use them.
// easily pick and map objects to dart objects
final Shoe oneShoe = pick(json, 'shoes', 0).letOrNull((p) => Shoe.fromPick(p));
// map list of picks to dart objects
final List<Shoe> shoes =
pick(json, 'shoes').asListOrEmpty((p) => Shoe.fromPick(p.required()));
Pick
now offers a new required()
method returning a RequiredPick
. It makes sure the picked value exists or crashes if it is null
. Because it can't be null
, RequiredPick
doesn't offer fallback methods like .asIntOrNull()
but only .asInt()
. This makes the API a bit easier to use for values you can't live without.
// use required() to crash if a object doesn't exist
final name = pick(json, 'shoes', 0, 'name').required().asString();
print(name); // Nike Zoom Fly 3
Note: Calling .asString()
directly on Pick
has been deprecated. You now have to call required()
first to convert the Pick
to a RequiredPick
or use a mapping method with fallbacks.
Ever got a Pick
/RequiredPick
and you wanted to pick even further. This is now possible with the call
method. Very useful in constructors when parsing methods.
factory Shoe.fromPick(RequiredPick pick) {
return Shoe(
id: pick('id').required().asString(),
name: pick('name').required().asString(),
manufacturer: pick('manufacturer').asStringOrNull(),
tags: pick('tags').asListOrEmpty(),
);
}
- Don't crash when selecting a out of range index from a
List
.asMap()
,.asMapOrNull()
and.asMapOrEmpty()
now consistently returnMap<dynamic, dynamic>
(wasMap<String, dynamic>
)
Also the lib has been converted to use static extension methods which were introduced in Dart 2.6
asMap
now expects the key type, defaults to dynamic
instead of String
-Pick.asMap(): Map<String, dynamic>
+Pick.asMap<T>(): Map<T, dynamic>
New API!
The old parse*
methods are now deprecated, but still work.
Replace them with the new pick(json, arg0-9...)
method.
- final name = parseJsonToString(json, 'shoes', 0, 'name');
+ final name = pick(json, 'shoes', 0, 'name').asString();
pick
returns a Pick
which offers a rich API to parse values.
.asString()
.asStringOrNull()
.asMap()
.asMapOrEmpty()
.asMapOrNull()
.asList()
.asListOrEmpty()
.asListOrNull()
.asBool()
.asBoolOrNull()
.asBoolOrTrue()
.asBoolOrFalse()
.asInt()
.asIntOrNull()
.asDouble()
.asDoubleOrNull()
.asDateTime()
.asDateTimeOrNull()
- pubspec description updated
- Initial version