cr-melon

Class based Http APIs in crystal

Melon

Class based Http APIs in crystal

Description

This is a proof of concept of creating a shard that allows people to create APIs with classes with the following fetaures:

  • DSL for building the APIs (get, put, post, etc...)
  • Inherit form an other APIs
  • Mounting other APIs at an endpoint
  • Inspectable routes and APIs for documentation

Example:

require "../src/melon"

USERS = [
  {name: "John Doe"},
  {name: "Someone Else"},
]

class Users < Melon::Api
  description "Users Endpoint"

  get description: "Get all users" do
    json USERS
  end
end

class Root < Melon::Api
  description "My Awesome API"

  get description: "Simple GET request." do
    ok "text/plain", "Hello there!"
  end

  post do
    ok "text/plain", "You have posted something."
  end

  mount Users, "users"
end

Melon.print_routes Root
Melon.listen Root, 8080

# Root - My Awesome API
# ----------------------------------------
# ├─ GET - /         # Simple GET request.
# ├─ POST - /
# └─┬─ API - /users  # Users Endpoint
#   └─ GET - /       # Get all users
# ----------------------------------------
# Listening on http://0.0.0.0:8080

Explanation

This implementation is heavily uses macros and a pattern matching to make it work:

  • macros create a subclass class Route%id < Route # Route__temp_25
  • macros create overloaded methods with the subclass def handle_route(id : Route__temp_25)
  • macros save the instances of these classes in a registry for access with metadata Registry[path] = Route%id.new "some metadata"
  • overloaded methods are run with the subclass from the matching route in the registry instance.handle_route(Registry[path])
  • a fallback implemention is needed for matching the base class def handle_route(id : Route) for compability (to actually compile) which will never be called
  • all created routes are inspectable in the registry

This behavior used can further explained with the following code:

# Define a class
class A
end

# Define a registry
REGISTRY = {} of String => A

# Define the which will have the DSL
class B
  # Define the DSL method
  macro make_print(key, text)
    # Create a sub class with a unique name
    class A%name < A
    end

    # Save an instance of that class in the registry
    REGISTRY[{{key}}] = A%name.new

    # Create an overloaded method that responds to that class only
    def handle_print(id : A%name)
      # Run things here
      puts {{text}}
    end
  end

  # Create a fallback method for compability
  def handle_print(id : A)
    puts "fallback"
  end

  # Have a method call the overloaded functions
  def print(key)
    handle_print REGISTRY[key]
  end

  # Actually make the overloaded methods
  make_print "a", "hello"
  make_print "b", "bye"
end

b = B.new
b.print "a" # "hello"
b.print "b" # "bye"
Repository

cr-melon

Owner
Statistic
  • 12
  • 0
  • 0
  • 0
  • 0
  • about 6 years ago
  • July 8, 2016
License

Links
Synced at

Tue, 21 Jan 2025 11:48:40 GMT

Languages