Bianchi

A DSL (Domain-Specific Language) and a minimalist framework in Ruby, tailored for USSD development. Structured around a menu page approach, Bianchi offers a comprehensive suite of methods and generators, streamlining the process of building USSD applications efficiently and easily. A basic test run with bianchi can be found here https://github.com/SydDaps/sinatra_bianchi_example

Installation

Add gem 'bianchi', '~> 0.1.1' to your gem file and run bundle install

Bianchi relies on Redis and requires the REDIS_URL environment variable to be set, pointing to your Redis instance. Ensure Redis is installed, and running, and that REDIS_URL is correctly configured before using Bianchi.

Getting Started

To initialize a new USSD project, generate a project directory using Bianchi's command-line interface. Use the following Ruby command:

bundle exec bianchi setup -p provider_name

Replace provider_name with your desired provider. Currently supported providers include: africastalking.

This command creates a ussd/engine.rb file in the project root directory. Here's a sample content of ussd/engine.rb:

module USSD
  class Engine
    def self.start(params)
      Bianchi::USSD::Engine.start(params, provider: :africastalking) do
        # e.g menu :main, options
      end
    end
  end
end

To define menus for your USSD applications' engine instance, use the following command:

menu :menu_name, options

For example, let's define our first menu, which serves as the initial menu of the application, and let's call this the main menu:

menu :main, initial: true

now that we have the initial menu up let's generate our pages for that menu using Bianchi's command-line interface. Use the following Ruby command:

command: bundle exec bianchi g menu menu_name page page_number
example: bianchi g menu main page 1

This creates a ussd/main_menu/page_1 file in the project root directory. Here's a sample content of the file:

module USSD
  module MainMenu
    class Page1 < Bianchi::USSD::Page
      def request; end

      def response; end
    end
  end
end

In the ussd/main_menu/page_1 file, the main application code goes into the request and response methods. Here's a sample code to illustrate the usage:

module USSD
  module MainMenu
    class Page1 < Bianchi::USSD::Page
      def request
        render_and_await(request_body)
      end

      def response
        case session_input_body
        when "1"
          redirect_to_greetings_menu_page_1
        when "2"
          redirect_to_repeat_menu_page_1
        else
          render_and_await("invalid input \n" + request_body)
        end
      end

      private

      def request_body
        <<~MSG
          Welcome
          1. Greetings
          2. Repeat my name
        MSG
      end
    end
  end
end

In this example, when a page is requested, you send some information, and the end user submits data for that request. The response method processes the response data and can move to a new page request, end the session, or send a response from there.

Page Renders

In Bianchi USSD applications, any method that starts with render sends data back to the user. Typically, we either send data and expect a response or send data and terminate the session. Here are the two main methods for rendering pages:

  • render_and_await(string): Responds to the user and expects the user to respond.

  • render_and_end(string): Responds to the user and terminates the session.

Here's how you can use these methods in Ruby:

render_and_await("Please select an option:")

This method sends the message "Please select an option:" to the user and awaits their response.

render_and_end("Thank you for using our service. Goodbye!")

This method sends the message "Thank you for using our service. Goodbye!" to the user and terminates the session.

Page Session

In Bianchi USSD applications, the session serves as an information tracker between all the pages and records all interactions between the user and the application. It contains valuable information necessary for the USSD application to function properly.

Here are the methods available within the page session:

  • session_params: Returns a hash containing all the data the provider sent.
  • session_input_body: Retrieves the user's current input.
  • session_mobile_number: Returns the current user's phone number dialing the code.
  • session_session_id: Provides the session's ID.
  • session.store: Accesses the current session's cache. Further details about this will be discussed later.

Page Store

In Bianchi USSD applications, the session.store serves as the main application cache, where data can be stored and accessed across all pages.

Here are the methods available within the page store:

  • session.store.set(key, value): Stores a key-value pair in the cache.
  • session.store.get(key): Retrieves the value associated with a given key from the cache.
  • session.store.all: Returns the entire cache as a hash.

Here's how you can use these methods to manage your application's cache:

# Storing data in the cache
session.store.set('user_id', 123)

# Retrieving data from the cache
user_id = session.store.get('user_id')

# Retrieving the entire cache
all_data = session.store.all

Page Redirects

In Bianchi USSD applications, any method that starts with redirect_to initiates another page request to be sent to the user. An example use case is when you want to transition to another menu page after processing the user's response.

For instance, in the snippet of the main menu page provided earlier, we use redirect_to_greetings_menu_page_1 when the user selects "1". This directs the application to the greetings menu page 1's request.

Here are the methods available within the page redirects:

  • redirect_to_menu_[menu_name]_page_[page_number]: Redirects to a specified menu using the menu_name and loads the page specified by page_number.
  • redirect_to_[next|previous]_page: Redirects to the next or previous page within the same menu.

Here's how you can use these methods within your Bianchi USSD application:

# Redirecting to a specific menu page
redirect_to_menu_greetings_page_1

# Redirecting to the next or previous page in the same menu
redirect_to_next_page
redirect_to_previous_page

Responding to the Provider

Once all processes are completed, the next step is to prepare the response to send to the provider. The provider prompt data is accessible via the engine instance's prompt_data attribute.

Here's an example of how you can retrieve the provider prompt data in your USSD application:

module USSD
  class Engine
    def self.start(params)
      Bianchi::USSD::Engine.start(params, provider: :africastalking) do
          menu :main, initial: true
          menu :greetings
          menu :repeat
      end
    end
  end
end

# Retrieving provider prompt data
provider_data = USSD::Engine.new(provider_params).prompt_data