mongodbflutterretrofitdiorestheart

Flutter restheart oid mapping


Using flutter dio and retrofit I run into problems with a restheart API.

How does one properly map the mongodb oid?

I've got a main.dart like this:

import 'dart:convert';
import 'package:dio/dio.dart';
import 'dart:developer';

import 'package:retrofit_experiment/utils/rest_client.dart';

var auth = 'Basic ' + base64Encode(utf8.encode('admin:secret'));

Future<void> main(List<String> args) async {
  final dio = Dio(); // Provide a dio instance
  dio.options.headers["authorization"] = auth;

  final client = RestClient(dio);

  final json = await client.getUser("61a94d306c573846b4c0f8bf");

  log(
    'User with id 61a94d306c573846b4c0f8bf',
    name: 'getUser',
    error: jsonEncode(json),
  );

  client.getUser("61a94d306c573846b4c0f8bf").then((it) => print(it.id));

}

a rest_client.dart:

import 'package:retrofit/retrofit.dart';
import 'package:dio/dio.dart';

import 'package:retrofit_experiment/model/user.dart';

part 'rest_client.g.dart';

@RestApi(baseUrl: "http://localhost:8080")
abstract class RestClient {
  factory RestClient(Dio dio, {String baseUrl}) = _RestClient;

  @GET("/users/{id}")
  Future<User> getUser(@Path("id") String id);
}

which generates:

// GENERATED CODE - DO NOT MODIFY BY HAND

part of 'rest_client.dart';

// **************************************************************************
// RetrofitGenerator
// **************************************************************************

class _RestClient implements RestClient {
  _RestClient(this._dio, {this.baseUrl}) {
    baseUrl ??= 'http://localhost:8080';
  }

  final Dio _dio;

  String? baseUrl;

  @override
  Future<User> getUser(id) async {
    const _extra = <String, dynamic>{};
    final queryParameters = <String, dynamic>{};
    final _data = <String, dynamic>{};
    final _result = await _dio.fetch<Map<String, dynamic>>(_setStreamType<User>(
        Options(method: 'GET', headers: <String, dynamic>{}, extra: _extra)
            .compose(_dio.options, '/users/$id',
                queryParameters: queryParameters, data: _data)
            .copyWith(baseUrl: baseUrl ?? _dio.options.baseUrl)));
    final value = User.fromJson(_result.data!);
    return value;
  }

  RequestOptions _setStreamType<T>(RequestOptions requestOptions) {
    if (T != dynamic &&
        !(requestOptions.responseType == ResponseType.bytes ||
            requestOptions.responseType == ResponseType.stream)) {
      if (T == String) {
        requestOptions.responseType = ResponseType.plain;
      } else {
        requestOptions.responseType = ResponseType.json;
      }
    }
    return requestOptions;
  }
}

and this user.dart:

import 'package:json_annotation/json_annotation.dart';
part 'user.g.dart';

@JsonSerializable()
class User {
  @JsonKey(name: '_id.\$oid')
  String? id;
  @JsonKey(name: 'first_name')
  String? firstName;
  @JsonKey(name: 'last_name')
  String? lastName;
  String? avatar;
  String? email;

  User({this.id, this.firstName, this.lastName, this.avatar, this.email});

  factory User.fromJson(Map<String, dynamic> json) => _$UserFromJson(json);
  Map<String, dynamic> toJson() => _$UserToJson(this);
}

which generates:

// GENERATED CODE - DO NOT MODIFY BY HAND

part of 'user.dart';

// **************************************************************************
// JsonSerializableGenerator
// **************************************************************************

User _$UserFromJson(Map<String, dynamic> json) {
  return User(
    id: json[r'_id.$oid'] as String?,
    firstName: json['first_name'] as String?,
    lastName: json['last_name'] as String?,
    avatar: json['avatar'] as String?,
    email: json['email'] as String?,
  );
}

Map<String, dynamic> _$UserToJson(User instance) => <String, dynamic>{
      r'_id.$oid': instance.id,
      'first_name': instance.firstName,
      'last_name': instance.lastName,
      'avatar': instance.avatar,
      'email': instance.email,
    };

I get the response data like this:

{"_id.$oid":null,"first_name":"Emma","last_name":"…faces/3-image.jpg","email":"emma.wong@reqres.in"}

Everything gets properly mapped but not the oid. What am I doing wrong?


Solution

  • I'm not familiar with Dart, however I suspect the error is that you are trying to get the _id as a String where it is an ObjectId.

    RESTHeart represents the ObjectId using the so called strict mode

    {
     "_id": { "$oid": "61a94d306c573846b4c0f8bf" }
    }
    

    You might want to look at https://github.com/mongo-dart/mongo_dart/issues/145 to have an example on how to parse an ObjectId from its string representation.