AppManager

Welcome to the new gem of Hulkapps.com ! An API wrapper of Hulkapps.com's AppManager portal.

Installation

  • Add following line to your application's Gemfile:
gem 'app_manager'
  • Execute following in terminal:
    bundle install
    rails g app_manager:install

this install command will mount routes in your rails route and will create a config file at config/initializers/app_manager.rb

    rails db:migrate

Configuration

  • Set Shopify API version to 2022-04 as app_manager is best compatible with this version.

  • Please map new fields (which added by app manager migration) in 'config.field_names' in below initializer file which is generated by install command. Please note: this is important step.

  • Please update blank fields in initializer file. Please copy 'uuid' from this file and paste in the following file and make sure that is unique per feature. After copying put your 'app name' next to 'uuid' column in this file.

AppManager.configure do |config|
  config.enable_caching = false  # Optional, True to enable app-manager api response caching, default is enabled from gem
  config.expires_in = 1.days  # Optional, Example: 30.seconds, 5.minutes or 2.days  Default caching is for 1.days from gem 
  config.app_url = '' # App URL like https://volumediscount.hulkapps.dev/ or #https://5044-2409-4052-209a-69da-9d7-925a-9418-a9c3.ngrok.io
  config.shopify_api_key = '' # Shopify api key of app
  config.shopify_api_version = ''  # Must be 2022-04 or latest
  config.shopify_table_name = 'shops'  #  Table name which is generated by shopify mostly it is 'shops'
  config.shopify_domain_field = 'shopify_domain' #shopify domain field
  config.plan_id_or_name_field = 'plan_id'
  config.shopify_app_slug = '' #Add your app slug here
  config.plan_page_route = '' #Add your app's plan page route here without first slash. Ex. "payment" 
  config.field_names = {
        'name' => 'shopify_domain', # demo-rahul-tiwari.myshopify.com
        'shopify_email' => 'email', # rahul.t@hulkapps.com
        'shopify_token' => 'shopify_token',
        'shopify_plan' => 'shopify_plan', # partner_test
        'plan_id' => 'plan_id', # 1. t
        'created_at' => 'created_at', # 2022-04-15 10:43:05
        'trial_activated_at' => 'trial_activated_at', # field name that stores trial start/activated date
        'grandfathered' => 'grandfathered',
        'total_trial_days' => '' #optional, put a trial days field from your shops table otherwise leave it blank
    }
  config.plan_features =  [
    {
        "uuid" => "b48a3a6c-c1fb-11ec-9d64-0242ac120002",
        "name" => "Features 1",
        "slug" => "feature-1",
        "description" => "Feature Description",
        "value_type" => "integer",
        "format" => "count",
        "display_order" => 1,
        "hidden_feature" => false,
        "group_order" => "1",
        "group" => "Group Name",
    },
    {
        "uuid" => "9f18f95a-bfaf-11ec-9d64-0242ac120002",
        "name" => "Features 2",
        "slug" => "feature-2",
        "description" => "Feature Description",
        "value_type" => "boolean",
        "format" => "percentage",
        "display_order" => 2,
        "hidden_feature" => false,
        "group_order" => "1",
        "group" => "Group Name",
    },
    {
        "uuid" => "9f190a26-bfaf-11ec-9d64-0242ac120002",
        "name" => "Features 3",
        "slug" => "feature-3",
        "description" => "Feature Description",
        "value_type" => "string",
        "format" => "string",
        "display_order" => 3,
        "hidden_feature" => false,
        "group_order" => "1",
        "group" => "Group Name",
    },
    {
        "uuid" => "9f191340-bfaf-11ec-9d64-0242ac12000",
        "name" => "Features 4",
        "slug" => "feature-4",
        "description" => "Feature Description",
        "value_type" => "array",
        "values" => [
            "val-1" => "val 1",
            "val-2" => "val 2",
            "val-3" => "val 3",
        ],
        "format" => "string",
        "display_order" => 4,
        "hidden_feature" => false,
        "group_order" => "1",
        "group" => "Group Name",
    }
]  
end
#Required, Values type : integer, boolean, string, array #
#Format: percentage, count, string
  • You must have to set ENV variable with key 'APP_MANAGER_ACCESS_TOKEN' in your application.yml or .env file like this:
    APP_MANAGER_ACCESS_TOKEN: 'XXXXXXXXXXXXXXXXXXX'
    APP_MANAGER_API_URL: 'https://XXXXXXXX.com'
  • App Manager provides a rake task that must needs to include in your existing app cron for every 10 minutes. Like if you are using whenever gem then you MUST need to include following rake take in your 'schedule.rb' (created by whenever gem)
every 10.minutes do
  rake 'sync:local_app_manager'
end

Usage

  • App Manager provides a helper module which needs to include in your 'shop' model something like below
class Shop < ActiveRecord::Base
  include AppManager::Model
end

and then you can use follwing methods with your shop objects.

@shop.has_plan # return true or false

@shop.plan_features # return array of plan features of your shop's plan 

@shop.has_feature('feature-1') # provide slug of feature which you set in your 'config.plan_features' in config/initializers/app_manager.rb
#return true/false

@shop.get_feature('feature-3') #  provide slug of feature and this returns value of feature which is set in portal. 

@shop.get_remaining_days # return integer based on trial activated date.

@shop.get_plan # return current plan hash

    if you pass @shop.get_plan(plan_id) it will return that particular plan hash like @shop.get_plan(311)

@shop.get_charge # return current charge hash from app manager with 'active_charge' & 'cancelled_charge' keys, it will return nil if current plan is free

@shop.cancel_charge # it will cancel charge in app manager and will return success to true if done from app manager

@shop.set_default_plan(plan_id=nil)  #if plan id is passed nil, then it will set the free(public and $0 price) plan_id in database and if passed it will set that plan_id in database.

@shop.get_plans_by_features(feature_slugs)  # pass feature_slugs as an array even if one slug and it will return plans array which have that slug found. Example: 

    @shop.get_plans_by_features(['quantity-based-discount'])
    @shop.get_plans_by_features(['quantity-based-discount','multiple-discount-types'])


    you can pass string argument 'exclude_current_plan' which will return all plans that matching features except current plan

    @shop.get_plans_by_features(['quantity-based-discount'],'exclude_current_plan')
    @shop.get_plans_by_features(['quantity-based-discount','multiple-discount-types'],'exclude_current_plan')


@shop.get_all_plans # return all app plans array

@shop.get_active_charge_app_manager # return active charge hash from app manager otherwise nil

@shop.update_app_manager_charge # return true if shop has shopify recurring charge but app manager doesn't then it update to app manager and return true otherwise nil. This method also accepts current shopify charge id, so you can pass it if you already. This will reduce the extra call like this @shop.update_app_manager_charge("xx23432434")

@shop.update_app_manager_charge_with_plan_id # same as update_app_manager_charge method, the only difference is that, it will not check the plan_id in the shop table is valid from app manager. This method also accepts two parameters, first is current shopify charge id and second is plan_id .It shop table doesn't have plan_id saved then you can pass in second parameter. @shop.update_app_manager_charge_with_plan_id("xx23432434",23)

@shop.get_current_shopify_charge # return active charge hash from shopify https://shopify.dev/docs/api/admin-graphql/2022-10/queries/currentAppInstallation otherwise nil

@shop.active_shopify_charge_id # return active shopify charge id if shop has any current shopify reccurring charge otherwise nil

@shop.cancel_current_shopify_charge # It will cancel charge in shopify if there is any, if success then it will cancel charge in app manager as well using (@shop.cancel_charge ) and will make plan_id and grandfathered field to false. It will clear cache as well.

@shop.active_shopfiy_charge_full #It will give shopify recurring charge graphql object with discounted price details

@shop.usage_charge_call(amount,description,shopify_charge_id #optional) #It will call usage charge API. Amount parameter must be present and should be float value. You can skip shopify_charge_id if you want to pass active shopify charge id. This method will automatically pull that, otherwise you can pass it. You have to pass descriptions as a second parameter.

  • Update app manager vue package in Rails+Vue apps & for for non-vue apps, update css or js from npm package, minimum version should be 2.4.4

  • Update app manager gem in your project's Gemfile

gem 'app_manager', '2.2.3'
  • In Rails project, in application.rb, add following line after require "rails/all" line
require 'app_manager/set_local_storage'
  • In route.rb file, add following route with constraints ruby condition = ->(request) { request.path_info.include?('/discount/') } plan_url = "https://admin.shopify.com/admin/apps/#{ENV['SHOPIFY_APP_SLUG']}/home/plan" constraints(condition) do get '/discount/*any', to: AppManager::SetLocalStorage.new(Rails.application, condition, plan_url) end
  • If updating gem from 1.6.1, then run these commands on rails root.

Note: In this command, Please do not overwrite app_manager.rb file so press n

rails g app_manager:install

This will add a new migration file in your db/app_manager directory

rails db:migrate

Extras

  • To view the app_manager ruby gem is working in your rails app you can use rails console and initialize app_manager instance like with App Manager Portal access:
ob = AppManager::Client.new(ENV['APP_MANAGER_ACCESS_TOKEN'])

To get banners use this command:

ob.get_banners
  • For Vue UI Library in Rails Project:

Please make sure you have following line in your route file otherwise add this manually.

mount AppManager::Engine, at: "/"

Development

After checking out the repo, run bin/setup to install dependencies. Then, run rake test to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.

To install this gem onto your local machine, run bundle exec rake install. To release a new version, update the version number in version.rb, and then run bundle exec rake release, which will create a git tag for the version, push git commits and the created tag, and push the .gem file to rubygems.org.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/app_manager. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the code of conduct.

Code of Conduct

Everyone interacting in the AppManager project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the code of conduct.