Skip to content

Conversation

@CLNMR
Copy link

@CLNMR CLNMR commented Oct 26, 2024

This PR adds a getParams method to the handler. This allows to extract the params of a URI from a request without actually handling the request in a middleware.

As far as I am aware, this was not possible in any other way before, as the routes are private in a Router.


  • I’ve reviewed the contributor guide and applied the relevant portions to this PR.
Contribution guidelines:

Note that many Dart repos have a weekly cadence for reviewing PRs - please allow for some latency before initial review feedback.

@google-cla
Copy link

google-cla bot commented Oct 26, 2024

Thanks for your pull request! It looks like this may be your first contribution to a Google open source project. Before we can look at your pull request, you'll need to sign a Contributor License Agreement (CLA).

View this failed invocation of the CLA check for more information.

For the most up to date status, view the checks section at the bottom of the pull request.

@kevmoo
Copy link
Member

kevmoo commented Oct 29, 2024

@CLNMR – add tests and an entry to the changelog, please!

@CLNMR
Copy link
Author

CLNMR commented Oct 31, 2024

@CLNMR – add tests and an entry to the changelog, please!

Sorry @kevmoo, I added them!

Co-authored-by: Kevin Moore <kevmoo@users.noreply.github.com>
@kevmoo
Copy link
Member

kevmoo commented Nov 1, 2024

@devoncarew @natebosch – thoughts here?

@kevmoo
Copy link
Member

kevmoo commented Nov 1, 2024

@CLNMR – need to change the pubspec version to match the changelog!


/// Get URL parameters captured by the [Router].
/// Returns `null` if no parameters are captured.
Map<String, String>? getParams(Request request) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we decide to land this, how about the name urlParameters.

https://dart.dev/effective-dart/design#avoid-starting-a-method-name-with-get

@natebosch
Copy link
Member

I think @jonasfj might have the most context on this package and whether this is a good solution.

I'm not sure I fully understand the problem. What is the use case where URL parameters must be read without handling the request?

Copy link
Member

@jonasfj jonasfj left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Functionality is already there:

https://pub.dev/documentation/shelf_router/latest/shelf_router/params.html

update, I missed this: 🤣

This allows to extract the params of a URI from a request without actually handling the request in a middleware.

Copy link
Member

@jonasfj jonasfj left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's kind of intentional that internals of this package are not public. It makes it possible to refactor the implementation, add new features without doing breaking changes.

We could make _routes public, but if you want fast moving APIs that can do everything, perhaps it better to build it as a package in the community.

What is the intended use case for this?

}
var params = route.match('/${request.url.path}');
if (params != null) {
return params;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Technically, the route is not necessarily matched.
The Handler for the route may return Router.routeNotFound, in which case the Router will try all subsequent routes.

See documentation for Router.routeNotFound.

It's kind of weird to look at matched parameters from different routes and mounted routers.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, that's true, but only when a route was matched and found are the params returned. Therefore you can deduce whether the router found a route or not (see also my comment below).

@CLNMR
Copy link
Author

CLNMR commented Nov 14, 2024

Hi, thanks for the feedback.

To expand a little on our use case:
We have several routers, each with a number of different routes.
Now, a lot of those routes (but not all!) have a workspaceId in their path. Currently, in each function that handles such a route, we must look up (and verify) the workspace using this id.
We would love to move this logic into a middleware, that populates the request with the workspace. BUT: This logic should only be used when <workspaceId> is part of the path!

This is where this feature comes in: Using this, we can give the list of routers to the middleware that parses the workspace. In this middleware, we can then call the proposed function (via null-aware assignment) so that we get the extracted parameters from the router that will later match the requested route. Then, we can run the initialisation logic for the workspace iff the workspaceId was extracted.

Here are the important parts of our current code using this feature:

server.dart

void main() async {
  final routers = [
    RecordApiService().router,
    CalendarApiService().router,
    SwaggerService().router,
    FlowApiService().router,
    BillingAccountApiService().router,
  ];

  final pipeline = Pipeline()
      .addMiddleware(logRequests())
      .addMiddleware(exceptionHandler)
      .addMiddleware(corsHeaders)
      .addMiddleware(**routeInspectorMiddleware**(routers))
      .addMiddleware(sessionControllerMiddleware)
      .addMiddleware(recordSpecMiddleware)
      .addMiddleware(recordMiddleware)
      .addMiddleware(flowSpecMiddleware)
      .addMiddleware(stripeCustomerMiddleware)
      .addHandler(routers.fold<Cascade>(Cascade(), (c, r) => c.add(r)).handler);

  await io.serve(pipeline, '0.0.0.0', int.parse(portStr));
}

route_inspector_middleware.dart

Middleware routeInspectorMiddleware(List<Router> routers) {
  return (Handler handler) {
    return (Request request) {
      for (final router in routers) {
        final params = router.getParams(request);
        if (params != null) {
          // Attach the matched route to the request context
          final updatedRequest = request.change(context: {'params': params});
          return handler(updatedRequest);
        }
      }
      return handler(request);
    };
  };
}

session_controller_middleware.dart

FutureOr<Response> Function(Request) sessionControllerMiddleware(
  Handler handler,
) =>
    (request) async {
      final workspaceId =
          (request.context['params'] as Map<String, dynamic>?)?['workspaceId'];

      if (workspaceId is String) {
        // do initialisation logic to get stateSnap
        final updatedRequest =
            request.change(context: {'stateSnap': stateSnap});
        return handler(updatedRequest);
      }
      return handler(request);
    };

Co-authored-by: Jonas Finnemann Jensen <jopsen@gmail.com>
@jonasfj
Copy link
Member

jonasfj commented Nov 15, 2024

Have you considered being more explicit, this is a bit scary because suddenly <workspaceId> will have all sorts of side effects. You can afaik do:

Suppose you have:

final router = Router();

router.get('/api/<workspaceId>/info', (Request request, String workspaceId) async {
  final workspace = await loadWorkspace(workspaceId);
  ...
});
router.get('/api/<workspaceId>/file/<id>', (Request request, String workspaceId, String id) async {
  final workspace = await loadWorkspace(workspaceId);
  ...
});
router.put('/api/<workspaceId>/file/<id>', (Request request, String workspaceId, String id) async {
  final workspace = await loadWorkspace(workspaceId);
  ...
});

IMO: This is not a bad setup at all, having to call loadWorkspace in every handler is annoying, but it works.

If you wanted smarter you could also do an extension method:

extension WorkspaceRequest on Request {
  Future<Workspace> get workspace {
    final workspaceId = params['workspaceId'];
    if (workspaceId == null) {
      throw StateError('Router for this handler does contain /<workspaceId>/');
    }
    return loadWorkspace(workspaceId);
  }
}

final router = Router();

router.get('/api/<workspaceId>/info', (Request request, String _) async {
  final workspace = await request.workspace;
  ...
});
router.get('/api/<workspaceId>/file/<id>', (Request request, String _, String id) async {
  final workspace = await request.workspace;
  ...
});
router.put('/api/<workspaceId>/file/<id>', (Request request, String _, String id) async {
  final workspace = await request.workspace;
  ...
});

Of course accessing request.workspace outside a handler for a route with /<workspaceId>/ in the route will throw.


If we should add more features to Router I think we should consider support for middleware.

extension WorkspaceRequest on Request {
  Workspace get workspace => context['workspace'] as Workspace;
}

final router = Router();

// NOTE: middlewares are not supported at the moment.
// But you can almost do it by returning Router.routeNotFound, you just can modify
// the request when doing this.
router.middleware('/api/<workspaceId>', (Handler handler, String workspaceId) async {
  final workspace = await loadWorkspace(workspaceId);
  return (request) {
    return handler(request.change(context: {'workspace': workspace}));
  }
});

router.get('/api/<workspaceId>/info', (Request request, String _) async {
  final workspace = await request.workspace;
  ...
});
router.get('/api/<workspaceId>/file/<id>', (Request request, String _, String id) async {
  final workspace = await request.workspace;
  ...
});
router.put('/api/<workspaceId>/file/<id>', (Request request, String _, String id) async {
  final workspace = await request.workspace;
  ...
});

Yes, you might need to add this middleware to all the different route patterns that contain /<workspaceId>/, but probably you have a limited number of such patterns -- much fewer than you have routes. And now it's all more explicit.


Just some quick ideas off the top of my head.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

4 participants