Networking
- Introduction
- Making HTTP requests
- Using an API Service
- Create an API Service
- Morphing JSON payloads to models
Introduction
Nylo makes networking in modern mobile applications simple. You can do GET, PUT, POST and DELETE requests via the base networking class.
Your API Services directory is located here app/networking/*
Fresh copies of Nylo will include a default API Service app/networking/api_service.dart
.
class ApiService extends BaseApiService {
ApiService({BuildContext? buildContext}) : super(buildContext);
@override
String get baseUrl => "https://jsonplaceholder.typicode.com";
@override
final interceptors = {
LoggingInterceptor: LoggingInterceptor()
};
Future<User?> fetchUsers() async {
return await network<User>(
request: (request) => request.get("/users"),
);
}
Variables you can override using the BaseApiService class.
-
baseUrl
- This is the base URL for the API, e.g. "https://jsonplaceholder.typicode.com". -
interceptors
- Here, you can add Dio interceptors. Learn more about interceptors here. -
useInterceptors
- You can set this to true or false. It will let the API Service know whether or not to use your interceptors.
Under the hood, the base networking class uses Dio, a powerful HTTP client.
Making HTTP requests
In your API Service, use the network
method to build your API request.
class ApiService extends BaseApiService {
ApiService({BuildContext? buildContext}) : super(buildContext);
@override
String get baseUrl => "https://jsonplaceholder.typicode.com";
Future<dynamic> fetchUsers() async {
return await network(
request: (request) {
// return request.get("/users"); // GET request
// return request.put("/users", data: {"user": "data"}); // PUT request
// return request.post("/users", data: {"user": "data"}); // POST request
// return request.delete("/users/1"); // DELETE request
return request.get("/users");
},
);
}
The request
argument is a Dio instance so you can call all the methods from that object.
Base Options
The BaseOptions
variable is highly configurable to allow you to modify how your Api Service should send your requests.
Inside your API Service, you can override your constructor.
class ApiService extends BaseApiService {
ApiService({BuildContext? buildContext}) : super(buildContext) {
baseOptions = BaseOptions(
receiveTimeout: 10000, // Timeout in milliseconds (10 seconds)
connectTimeout: 5000 // Timeout in milliseconds (5 seconds)
);
}
...
Click here to view all the base options you can set.
Adding Headers
You can add headers to your requests either via your baseOptions variable, on each request or an interceptor.
Here's the most simple way to add headers to a request.
...
Future<dynamic> fetchUsers() async {
return await network<dynamic>(
request: (request) {
request.options.headers = {
"Authorization": "Bearer $token"
};
return request.get("/users");
},
);
}
Interceptors
If you're new to interceptors, don't worry. They're a new concept for managing how your HTTP requests are sent.
Put in simple terms. An 'interceptor' will intercept the request, allowing you to modify the request before it's sent, handle the response after it completes and also what happens if there's an error.
Nylo allows you to add new interceptors to your API Services like in the below example.
class ApiService extends BaseApiService {
ApiService({BuildContext? buildContext}) : super(buildContext);
...
@override
final interceptors = {
LoggingInterceptor: LoggingInterceptor(),
// Add more interceptors for the API Service
// BearerAuthInterceptor: BearerAuthInterceptor(),
};
...
Let's take a look at an interceptor.
Example of a custom interceptor.
import 'package:nylo_framework/nylo_framework.dart';
class CustomInterceptor extends Interceptor {
@override
void onRequest(RequestOptions options, RequestInterceptorHandler handler) {
// options - modify the RequestOptions before the request
return super.onRequest(options, handler);
}
@override
void onResponse(Response response, ResponseInterceptorHandler handler) {
// response - handle/manipulate the Response object
handler.next(response);
}
@override
void onError(DioError error, ErrorInterceptorHandler handler) {
// error - handle the DioError object
handler.next(err);
}
}
Fresh copies on Nylo will include a app/networking/dio/intecetors/*
directory.
Inside this folder, you will find a LoggingInterceptor
class.
File: app/networking/dio/intecetors/logging_interceptor.dart
import 'dart:developer';
import 'package:nylo_framework/nylo_framework.dart';
class LoggingInterceptor extends Interceptor {
@override
void onRequest(RequestOptions options, RequestInterceptorHandler handler) {
if (getEnv('APP_DEBUG') == true) {
print('REQUEST[${options.method}] => PATH: ${options.path}');
}
return super.onRequest(options, handler);
}
@override
void onResponse(Response response, ResponseInterceptorHandler handler) {
if (getEnv('APP_DEBUG') == true) {
print(
'RESPONSE[${response.statusCode}] => PATH: ${response.requestOptions
.path}');
print('DATA: ${response.requestOptions.path}');
log(response.data.toString());
}
handler.next(response);
}
@override
void onError(DioError err, ErrorInterceptorHandler handler) {
if (getEnv('APP_DEBUG') == true) {
print('ERROR[${err.response?.statusCode}] => PATH: ${err.requestOptions
.path}');
}
handler.next(err);
}
}
Understanding the network helper
The network
helper provides us with a way to make HTTP requests from our application.
The helper method can be accessed when using an API Service in Nylo.
class ApiService extends BaseApiService {
...
Future<dynamic> fetchUsers() async {
return await network(
request: (request) {
// return request.get("/users"); // GET request
// return request.put("/users", data: {"user": "data"}); // PUT request
// return request.post("/users", data: {"user": "data"}); // POST request
// return request.delete("/users/1"); // DELETE request
return request.get("/users");
},
);
}
Return Types
There are two ways to handle the response from an HTTP request. Let's take a look at both in action, there's no right or wrong way to do this.
Using Model Decoders
Model Decoders are a new concept introduced in Nylo v3.x.
They make it easy to return your objects like in the below example.
class ApiService extends BaseApiService {
...
Future<User?> fetchUser() async {
return await network<User>(
request: (request) => request.get("/users/1"),
);
}
File: config/decoders.dart
final modelDecoders = {
User: (data) => User.fromJson(data), // add your model and handle the return of the object
// ...
};
The data
parameter will contain the HTTP response body.
Learn more about decoders here
Using handleSuccess
The handleSuccess: (Response response) {}
argument can be used to return a value from the HTTP body.
This method is only called if the HTTP response has a status code equal to 200.
Here's an example below.
class ApiService extends BaseApiService {
...
// Example: returning an Object
Future<User?> findUser() async {
return await network(
request: (request) => request.get("/users/1"),
handleSuccess: (Response response) { // response - Dio Response object
dynamic data = response.data;
return User.fromJson(data);
}
);
}
// Example: returning a String
Future<String?> findMessage() async {
return await network(
request: (request) => request.get("/message/1"),
handleSuccess: (Response response) { // response - Dio Response object
dynamic data = response.data;
if (data['name'] == 'Anthony') {
return "It's Anthony";
}
return "Hello world";
}
);
}
// Example: returning a bool
Future<bool?> updateUser() async {
return await network(
request: (request) => request.put("/user/1", data: {"name": "Anthony"}),
handleSuccess: (Response response) { // response - Dio Response object
dynamic data = response.data;
if (data['status'] == 'OK') {
return true;
}
return false;
}
);
}
Using handleFailure
The handleFailure
method will be called if the HTTP response returns a status code not equal to 200.
You can provide the network helper with the handleFailure: (DioError dioError) {}
argument and then handle the response in the function.
Here's an example for how it works.
class ApiService extends BaseApiService {
...
// Example: returning an Object
Future<User?> findUser() async {
return await network(
request: (request) => request.get("/users/1"),
handleFailure: (DioError dioError) { // response - DioError object
dynamic data = response.data;
// Handle the response
return null;
}
);
}
}
Using an API Service
When you need to call an API from a widget there are two different approaches in Nylo.
- You can create a new instance of the API Service and then call the method you want to use like in the below example.
class _MyHomePageState extends NyState<MyHomePage> {
ApiService _apiService = ApiService();
@override
init() async {
List<User>? users = await _apiService.fetchUsers();
print(users); // List<User>? instance
...
- Use the
api
helper, this method is shorter and works by using yourapiDecoders
variable inside config/decoders.dart. Learn more about decoders here.
class _MyHomePageState extends NyState<MyHomePage> {
@override
init() async {
User? user = await api<ApiService>((request) => request.fetchUser());
print(user); // User? instance
...
Using the api helper also allows you to handle UI feedback to your users if the request isn't successful.
To do this, add the context
parameter to the api
helper, example below.
// Your Widget
...
User _user = User();
@override
Widget build(BuildContext context) {
return Scaffold(
body: MaterialButton(
onPressed: () {
_sendFriendRequest(_user);
},
child: Text("Send Friend Request"),
);
);
}
_sendFriendRequest(User user) async {
bool? successful = await api<ApiService>(
(request) => request.sendFriendRequest(user), context: context
);
}
...
// Your API Service
class ApiService extends BaseApiService {
...
// Add this
displayError(DioError dioError, BuildContext context) {
showToastNotification(context, title: 'Oops!', description: dioError.message);
// or display the error however you want
}
}
displayError
- If an error occurs with the request (e.g. 500 status code), you can instantly give your users feedback via a toast notification.
Create an API Service
To create more api_services, e.g. a user_api_service
, use the command below.
flutter pub run nylo_framework:main make:api_service user
You can also create an API Service for a model with the --model="User"
option.
flutter pub run nylo_framework:main make:api_service user --model="User"
This will create an API Service with the following methods.
class UserApiService extends BaseApiService {
...
/// Return a list of users
Future<List<User>?> fetchAll({dynamic query}) async {
return await network<List<User>>(
request: (request) => request.get("/endpoint-path", queryParameters: query),
);
}
/// Find a User
Future<User?> find({required int id}) async {
return await network<User>(
request: (request) => request.get("/endpoint-path/$id"),
);
}
/// Create a User
Future<User?> create({required dynamic data}) async {
return await network<User>(
request: (request) => request.post("/endpoint-path", data: data),
);
}
/// Update a User
Future<User?> update({dynamic query}) async {
return await network<User>(
request: (request) => request.put("/endpoint-path", queryParameters: query),
);
}
/// Delete a User
Future<bool?> delete({required int id}) async {
return await network<bool>(
request: (request) => request.delete("/endpoint-path/$id"),
);
}
}
Morphing JSON payloads to models
You can automatically decode JSON payloads from your API requests to models using decoders.
Here is an API request that uses Nylo's implementation of decoders.
class ApiService extends BaseApiService {
ApiService({BuildContext? buildContext}) : super(buildContext);
Future<User?> fetchUsers() async {
return await network<User>(
request: (request) => request.get("/users"),
);
}
}
The fetchUsers
method will handle the JSON conversion to the model representation using generics.
You will first need to add your model to your config/decoders.dart
file like the below.
/// file 'config/decoders.dart'
final modelDecoders = {
List<User>: (data) => List.from(data).map((json) => User.fromJson(json)).toList(),
User: (data) => User.fromJson(data),
// ...
};
Here you can provide the different representations of the Model, e.g. to object or list<Model> like the above.
The data argument in the decoder will contain the body payload from the API request.
To get started with decoders, check out this section of the documentation.