~singpolyma/sgx-jmp

ref: 48adae14cdd37c6b74b5b04bae279823fa85fd43 sgx-jmp/lib/call_attempt.rb -rw-r--r-- 2.3 KiB
48adae14Stephen Paul Weber Try auto top up / low balance notify when not enough balance for a call 9 months ago
                                                                                
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
# frozen_string_literal: true

require "value_semantics/monkey_patched"

require_relative "low_balance"

class CallAttempt
	EXPENSIVE_ROUTE = {
		"usd_beta_unlimited-v20210223" => 0.9,
		"cad_beta_unlimited-v20210223" => 1.1
	}.freeze

	def self.for(customer, other_tel, rate, usage, direction:, **kwargs)
		kwargs.merge!(direction: direction)
		included_credit = [customer.minute_limit.to_d - usage, 0].max
		if !rate || rate >= EXPENSIVE_ROUTE.fetch(customer.plan_name, 0.1)
			Unsupported.new(direction: direction)
		elsif included_credit + customer.balance < rate * 10
			NoBalance.for(customer, other_tel, rate, usage, **kwargs)
		else
			for_ask_or_go(customer, other_tel, rate, usage, **kwargs)
		end
	end

	def self.for_ask_or_go(customer, otel, rate, usage, digits: nil, **kwargs)
		can_use = customer.minute_limit.to_d + customer.monthly_overage_limit
		if digits != "1" && can_use - usage < rate * 10
			AtLimit.new(**kwargs.slice(:direction, :call_id))
		else
			new(from: customer.registered?.phone, to: otel, **kwargs)
		end
	end

	value_semantics do
		from(/\A\+\d+\Z/)
		to(/\A\+\d+\Z/)
		call_id String
		direction Either(:inbound, :outbound)
	end

	def to_render
		["#{direction}/connect", { locals: to_h }]
	end

	def create_call(fwd, *args, &block)
		fwd.create_call(*args, &block)
	end

	class Unsupported
		value_semantics do
			direction Either(:inbound, :outbound)
		end

		def to_render
			["#{direction}/unsupported"]
		end

		def create_call(*); end
	end

	class NoBalance
		def self.for(customer, other_tel, rate, usage, direction:, **kwargs)
			LowBalance.for(customer).then(&:notify!).then do |amount|
				if amount&.positive?
					CallAttempt.for(
						customer.with_balance(customer.balance + amount),
						other_tel, rate, usage, direction: direction, **kwargs
					)
				else
					NoBalance.new(balance: customer.balance, direction: direction)
				end
			end
		end

		value_semantics do
			balance Numeric
			direction Either(:inbound, :outbound)
		end

		def to_render
			["#{direction}/no_balance", { locals: to_h }]
		end

		def create_call(*); end
	end

	class AtLimit
		value_semantics do
			call_id String
			direction Either(:inbound, :outbound)
		end

		def to_render
			["#{direction}/at_limit", { locals: to_h }]
		end

		def create_call(fwd, *args, &block)
			fwd.create_call(*args, &block)
		end
	end
end