blob: 7644653d52aa4b53ae4d9688f185afb19031c328 [file] [log] [blame]
// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
part of protobuf;
/// Returns the error message for an invalid field value,
/// or null if it's valid.
///
/// For enums, group, and message fields, this check is only approximate,
/// because the exact type isn't included in [fieldType].
String _getFieldError(int fieldType, var value) {
switch (PbFieldType._baseType(fieldType)) {
case PbFieldType._BOOL_BIT:
if (value is! bool) return 'not type bool';
return null;
case PbFieldType._BYTES_BIT:
if (value is! List) return 'not List';
return null;
case PbFieldType._STRING_BIT:
if (value is! String) return 'not type String';
return null;
case PbFieldType._FLOAT_BIT:
if (value is! double) return 'not type double';
if (!_isFloat32(value)) return 'out of range for float';
return null;
case PbFieldType._DOUBLE_BIT:
if (value is! double) return 'not type double';
return null;
case PbFieldType._ENUM_BIT:
if (value is! ProtobufEnum) return 'not type ProtobufEnum';
return null;
case PbFieldType._INT32_BIT:
case PbFieldType._SINT32_BIT:
case PbFieldType._SFIXED32_BIT:
if (value is! int) return 'not type int';
if (!_isSigned32(value)) return 'out of range for signed 32-bit int';
return null;
case PbFieldType._UINT32_BIT:
case PbFieldType._FIXED32_BIT:
if (value is! int) return 'not type int';
if (!_isUnsigned32(value)) return 'out of range for unsigned 32-bit int';
return null;
case PbFieldType._INT64_BIT:
case PbFieldType._SINT64_BIT:
case PbFieldType._UINT64_BIT:
case PbFieldType._FIXED64_BIT:
case PbFieldType._SFIXED64_BIT:
// We always use the full range of the same Dart type.
// It's up to the caller to treat the Int64 as signed or unsigned.
// See: https://github.com/dart-lang/protobuf/issues/44
if (value is! Int64) return 'not Int64';
return null;
case PbFieldType._GROUP_BIT:
case PbFieldType._MESSAGE_BIT:
if (value is! GeneratedMessage) return 'not a GeneratedMessage';
return null;
default:
return 'field has unknown type $fieldType';
}
}
// entry points for generated code
// generated checkItem for message, group, enum calls this
void checkItemFailed(val, String className) {
throw new ArgumentError('Value ($val) is not an instance of ${className}');
}
/// Returns a function for validating items in a repeated field.
///
/// For enum, group, and message fields, the check is only approximate,
/// because the exact type isn't included in [fieldType].
CheckFunc getCheckFunction(int fieldType) {
switch (fieldType & ~0x7) {
case PbFieldType._BOOL_BIT:
return _checkBool;
case PbFieldType._BYTES_BIT:
return _checkBytes;
case PbFieldType._STRING_BIT:
return _checkString;
case PbFieldType._FLOAT_BIT:
return _checkFloat;
case PbFieldType._DOUBLE_BIT:
return _checkDouble;
case PbFieldType._INT32_BIT:
case PbFieldType._SINT32_BIT:
case PbFieldType._SFIXED32_BIT:
return _checkSigned32;
case PbFieldType._UINT32_BIT:
case PbFieldType._FIXED32_BIT:
return _checkUnsigned32;
case PbFieldType._INT64_BIT:
case PbFieldType._SINT64_BIT:
case PbFieldType._SFIXED64_BIT:
case PbFieldType._UINT64_BIT:
case PbFieldType._FIXED64_BIT:
// We always use the full range of the same Dart type.
// It's up to the caller to treat the Int64 as signed or unsigned.
// See: https://github.com/dart-lang/protobuf/issues/44
return _checkAnyInt64;
case PbFieldType._ENUM_BIT:
return _checkAnyEnum;
case PbFieldType._GROUP_BIT:
case PbFieldType._MESSAGE_BIT:
return _checkAnyMessage;
}
throw new ArgumentError('check function not implemented: ${fieldType}');
}
// check functions for repeated fields
void _checkNotNull(Object val) {
if (val == null) {
throw new ArgumentError("Can't add a null to a repeated field");
}
}
void _checkBool(Object val) {
if (val is! bool) throw _createFieldTypeError(val, 'a bool');
}
void _checkBytes(Object val) {
if (val is! List<int>) throw _createFieldTypeError(val, 'a List<int>');
}
void _checkString(Object val) {
if (val is! String) throw _createFieldTypeError(val, 'a String');
}
void _checkFloat(Object val) {
_checkDouble(val);
if (!_isFloat32(val)) throw _createFieldRangeError(val, 'a float');
}
void _checkDouble(Object val) {
if (val is! double) throw _createFieldTypeError(val, 'a double');
}
void _checkInt(Object val) {
if (val is! int) throw _createFieldTypeError(val, 'an int');
}
void _checkSigned32(Object val) {
_checkInt(val);
if (!_isSigned32(val)) throw _createFieldRangeError(val, 'a signed int32');
}
void _checkUnsigned32(Object val) {
_checkInt(val);
if (!_isUnsigned32(val)) {
throw _createFieldRangeError(val, 'an unsigned int32');
}
}
void _checkAnyInt64(Object val) {
if (val is! Int64) throw _createFieldTypeError(val, 'an Int64');
}
void _checkAnyEnum(Object val) {
if (val is! ProtobufEnum) throw _createFieldTypeError(val, 'a ProtobufEnum');
}
void _checkAnyMessage(Object val) {
if (val is! GeneratedMessage) {
throw _createFieldTypeError(val, 'a GeneratedMessage');
}
}
ArgumentError _createFieldTypeError(val, String wantedType) =>
new ArgumentError('Value ($val) is not ${wantedType}');
RangeError _createFieldRangeError(val, String wantedType) =>
new RangeError('Value ($val) is not ${wantedType}');
bool _isSigned32(int value) => (-2147483648 <= value) && (value <= 2147483647);
bool _isUnsigned32(int value) => (0 <= value) && (value <= 4294967295);
bool _isFloat32(double value) =>
value.isNaN ||
value.isInfinite ||
(-3.4028234663852886E38 <= value) && (value <= 3.4028234663852886E38);