Result Controller Library
A robust library for functional error handling in Dart and Flutter that provides a safe way to manage operations that may fail. This library implements the Result/Either pattern for elegant error handling.
Features
- Functional Error Handling: Result<T, E>pattern with type safety
- Comprehensive API Error Management: Specialized tools for API responses
- Error Context Preservation: Maintains stack traces and original errors
- Chainable Operations: Fluent API for sequential operations
- Flexible Error Transformation: Conversion between error types
- Async Support: Handling of synchronous and asynchronous operations
- JSON Processing Utilities: Safe JSON data parsing and transformation
Installation
Add to your pubspec.yaml:
dependencies:
  result_controller: ^1.3.0
🚨 Recommended: Enable Lint Rules
For the best development experience, copy our recommended lint configuration:
# Copy recommended linting rules to your project
cp node_modules/result_controller/recommended_analysis_options.yaml analysis_options.yaml
Or manually add to your analysis_options.yaml:
include: package:lints/recommended.yaml
analyzer:
  errors:
    avoid_returning_null: error              # Use Ok() instead of null
    avoid_returning_null_for_future: error   # Use Future<Result<T,E>>
    avoid_catching_errors: warning           # Use Result.trySync()
linter:
  rules:
    - prefer_const_constructors             # Use const Ok() and const Err()
    - prefer_final_locals                   # Use final for Result variables
    - public_member_api_docs               # Document Result-returning methods
These rules will help your IDE guide you toward correct Result Controller usage!
Basic Usage
Base Result Type
The foundation of this library is the Result<T, E> type that represents either a successful value of type T or an error of type E.
import 'package:result_controller/result_controller.dart';
// Safe division function that cannot throw exceptions
Result<double, String> divideNumbers(int a, int b) {
  if (b == 0) {
    return Err('Cannot divide by zero');
  }
  return Ok(a / b);
}
void main() {
  final result = divideNumbers(10, 2);
  
  // Handle success and error cases
  result.when(
    ok: (value) => print('Result: $value'),  // Prints: Result: 5.0
    err: (error) => print('Error: $error'),
  );
  
  // Or extract the value directly (throws if it's an error)
  try {
    final value = result.data;
    print('Value: $value');  // Prints: Value: 5.0
  } catch (e) {
    print('Error accessing value: $e');
  }
}
Wrapping Exceptions
Use the trySync method to safely execute code that might throw exceptions:
// This function safely parses an integer
Result<int, ResultError> parseInteger(String input) {
  return Result.trySync(() => int.parse(input));
}
// Usage
final result = parseInteger('42');
final invalidResult = parseInteger('not a number');
result.when(
  ok: (value) => print('Parsed: $value'),  // Prints: Parsed: 42
  err: (error) => print('Parse error: $error'),
);
invalidResult.when(
  ok: (value) => print('Parsed: $value'),
  err: (error) => print('Parse error: $error'),  // Prints details about the FormatException
);
Transforming Results
Use map to transform the success value:
Result<int, String> getNumber() => Ok(10);
final stringResult = getNumber().map((number) => 'Number: $number');
stringResult.when(
  ok: (value) => print(value),  // Prints: Number: 10
  err: (error) => print('Error: $error'),
);
API Response Handling
ApiResponse Class
The ApiResponse class represents an API operation response, containing data, status code, and possible error information.
import 'package:result_controller/result_controller.dart';
// Create a successful response
final successResponse = ApiResponse.ok(
  {'id': '123', 'name': 'John Doe'},
  statusCode: 200,
  headers: {'Content-Type': 'application/json'},
);
// Create an error response
final errorResponse = ApiResponse.err(
  ApiErr(
    exception: Exception('Network timeout'),
    stackTrace: StackTrace.current,
    message: HttpMessage(
      title: 'Connection Error',
      details: 'Could not connect to server',
    ),
  ),
  statusCode: 503,
  headers: {},
);
Processing API Responses with when()
Use the when() method to handle success and error cases:
ApiResponse response = await apiClient.get('/users/123');
final userName = response.when(
  ok: (data) => data['name'] as String,
  err: (error) => 'Unknown User',
);
print('User name: $userName');
Processing Lists with whenList()
The whenList() method is specifically designed for API responses containing lists of objects:
ApiResponse response = await apiClient.get('/users');
final users = response.whenList(
  ok: (userList) => userList.map((userData) => User.fromJson(userData)).toList(),
  err: (error) {
    logError('Error fetching users', error);
    return <User>[]; // Return empty list on error
  },
);
// Now we have a typed list of User objects
for (final user in users) {
  print('User: ${user.name}');
}
Processing Typed Lists with whenListType()
Use whenListType() when working with primitive type lists or mixed content:
ApiResponse response = await apiClient.get('/user/scores');
// Extract a list of integers from the response
final scores = response.whenListType<List<int>, int>(
  ok: (scoreList) => scoreList,  // Already have the correct type
  err: (error) => <int>[],
  filterNulls: true,  // Remove null values from the list
);
final average = scores.isEmpty ? 0 : scores.reduce((a, b) => a + b) / scores.length;
print('Average score: $average');
Processing Dynamic JSON Maps with whenJsonListMap()
For handling complex and dynamic JSON structures:
ApiResponse response = await apiClient.get('/configurations');
final configurations = response.whenJsonListMap(
  ok: (configList) => configList.map((config) {
    return UserConfiguration(
      id: config['id'],
      settings: config['settings'] ?? {},
    );
  }).toList(),
  err: (error) => <UserConfiguration>[],
);
// Process configurations
for (final config in configurations) {
  applyConfiguration(config);
}
Converting ApiResponse to ApiResult
For more functional processing, convert to ApiResult:
ApiResponse response = await apiClient.get('/users/123');
// Convert to ApiResult
ApiResult<User> result = response.toResult(User.fromJson);
// Process with functional style
result.when(
  ok: (user) => displayUserProfile(user),
  err: (error) => showErrorMessage(error.message?.details ?? 'Unknown error'),
);
Converting List Responses
For list responses:
ApiResponse response = await apiClient.get('/posts');
// Convert list response to ApiResult
ApiResult<List<Post>> result = response.toListResult(
  (items) => items.map((item) => Post.fromJson(item)).toList(),
);
result.when(
  ok: (posts) => displayPosts(posts),
  err: (error) => showErrorMessage('Could not load posts'),
);
Advanced Features
Chaining Operations
Use flatMap to chain operations that might fail:
Future<Result<User, ApiErr>> getUser(String id) async {
  // Implementation details...
}
Future<Result<List<Post>, ApiErr>> getUserPosts(User user) async {
  // Implementation details...
}
// Chain operations
final postsResult = await getUser('123').flatMap((user) => getUserPosts(user));
postsResult.when(
  ok: (posts) => displayPosts(posts),
  err: (error) => showErrorMessage(error.message?.details ?? 'Unknown error'),
);
Error Recovery
Use recover to handle errors and potentially recover from them:
Future<Result<List<Post>, ApiErr>> getPosts() async {
  // Implementation details...
}
// Try to get from network, fall back to cache on error
final posts = await getPosts().recover((error) {
  if (error.statusCode == 503) {
    // Network unavailable, try to load from cache
    return loadPostsFromCache();
  }
  // Propagate any other error
  return Err(error);
});
Error Transformation
Convert between error types with mapError:
Result<User, ApiErr> fetchResult = await getUser('123');
// Convert API errors to user-friendly messages
Result<User, String> userResult = fetchResult.mapError((apiErr) {
  if (apiErr.statusCode == 404) {
    return 'User not found';
  } else if (apiErr.statusCode == 401) {
    return 'You need to log in first';
  }
  return 'An unexpected error occurred';
});
Collection Operations
Special extensions for working with collections:
// Filter a list result
Result<List<Post>, ApiErr> postsResult = await getPosts();
Result<List<Post>, ApiErr> recentPosts = postsResult.filter((post) => 
  post.date.isAfter(DateTime.now().subtract(Duration(days: 7)))
);
// Transform each element in a list result
Result<List<Post>, ApiErr> postsResult = await getPosts();
Result<List<String>, ApiErr> postTitles = postsResult.mapEach((post) => post.title);
Async Error Handling
Safe execution of async operations:
Future<void> performAsyncOperation() async {
  // Safely execute async code that might throw
  final result = await Result.tryAsync(() async {
    final response = await http.get(Uri.parse('https://api.example.com/data'));
    if (response.statusCode != 200) {
      throw Exception('Error loading data: ${response.statusCode}');
    }
    return json.decode(response.body);
  });
  
  result.when(
    ok: (data) => processData(data),
    err: (error) => showError(error),
  );
}
Custom Error Mapping
Transform exceptions to domain-specific errors:
Future<Result<User, UserError>> getUser(String id) async {
  return Result.tryAsyncMap(
    () async {
      final response = await http.get(Uri.parse('https://api.example.com/users/$id'));
      if (response.statusCode != 200) {
        throw HttpException('Error with status: ${response.statusCode}');
      }
      return User.fromJson(json.decode(response.body));
    },
    (error, stackTrace) {
      if (error is HttpException) {
        return UserError.network('Connection error: $error');
      } else if (error is FormatException) {
        return UserError.parsing('Invalid data format: $error');
      }
      return UserError.unknown('Unexpected error: $error');
    },
  );
}
Complete API Client Example
Here's a more complete example showing how to use Result Controller in an API client:
class ApiClient {
  final http.Client _client;
  final String _baseUrl;
  
  ApiClient(this._client, this._baseUrl);
  
  Future<ApiResponse> get(Params params) async {
    return Result.tryAsyncMap<ApiResponse, ApiResponse>(
      () async {
        final url = Uri.parse('$_baseUrl/${params.path}');
        final response = await _client.get(
          url,
          headers: params.header,
        );
        
        if (response.statusCode >= 200 && response.statusCode < 300) {
          return ApiResponse.ok(
            response.body,
            statusCode: response.statusCode,
            headers: response.headers,
          );
        } else {
          // Parse error response
          HttpMessage? errorMessage;
          try {
            final errorData = json.decode(response.body);
            errorMessage = HttpMessage.fromJson(errorData);
          } catch (_) {
            errorMessage = HttpMessage(
              title: 'HTTP Error',
              details: 'Request failed with status: ${response.statusCode}',
            );
          }
          
          throw HttpErr(
            exception: Exception('HTTP Error ${response.statusCode}'),
            stackTrace: StackTrace.current,
            message: errorMessage,
          );
        }
      },
      (error, stackTrace) {
        // Convert any exception to a failure response
        return ApiResponse.err(
          ApiErr(
            exception: error,
            stackTrace: stackTrace,
            message: HttpMessage.fromException(error),
          ),
        );
      },
    );
  }
  
  // Similar implementations for post, put, delete, etc.
}
// Usage
Future<ApiResult<User>> getUser(String id) async {
  final response = await apiClient.get(
    Params(path: 'users/$id', header: {'Authorization': 'Bearer $token'}),
  );
  
  return response.toResult(User.fromJson);
}
// Using the API client
void main() async {
  final userResult = await getUser('123');
  
  userResult.when(
    ok: (user) => print('User: ${user.name}'),
    err: (error) => print('Error: ${error.message?.details}'),
  );
}
Error Handling Strategy
The Result Controller library promotes a functional approach to error handling:
- Explicit Error Types: All possible errors are represented in the return type
- No Surprise Exceptions: Operations return errors instead of throwing exceptions
- Error Context Preservation: Maintains stack traces and original errors
- Composable Operations: Chain operations with appropriate error handling
- Typed Error Handling: Different error types for different contexts
Contributing
Contributions are welcome! If you have ideas for new features or improvements, please open an issue or submit a pull request.
- Fork the repository.
- Create a new branch (git checkout -b feature/new-feature).
- Commit your changes (git commit -am 'Add new feature').
- Push to the branch (git push origin feature/new-feature).
- Open a pull request.
License
This project is licensed under the MIT License - see the LICENSE file for details.