test_helper.rb

  1# frozen_string_literal: true
  2
  3require "simplecov"
  4SimpleCov.start do
  5	add_filter "/test/"
  6	enable_coverage :branch
  7end
  8
  9require "em_promise"
 10require "fiber"
 11require "minitest/autorun"
 12require "rantly/minitest_extensions"
 13require "sentry-ruby"
 14require "webmock/minitest"
 15begin
 16	require "pry-rescue/minitest"
 17	require "pry-reload"
 18
 19	module Minitest
 20		class Test
 21			alias old_capture_exceptions capture_exceptions
 22			def capture_exceptions
 23				old_capture_exceptions do
 24					yield
 25				rescue Minitest::Skip => e
 26					failures << e
 27				end
 28			end
 29		end
 30	end
 31rescue LoadError
 32	# Just helpers for dev, no big deal if missing
 33	nil
 34end
 35
 36require "backend_sgx"
 37require "tel_selections"
 38
 39$VERBOSE = nil
 40Sentry.init
 41
 42def customer(
 43	customer_id="test",
 44	plan_name: nil,
 45	jid: Blather::JID.new("#{customer_id}@example.net"),
 46	expires_at: Time.now,
 47	auto_top_up_amount: 0,
 48	**kwargs
 49)
 50	plan = CustomerPlan.for(
 51		customer_id,
 52		plan_name: plan_name,
 53		expires_at: expires_at,
 54		auto_top_up_amount: auto_top_up_amount
 55	)
 56	Customer.new(customer_id, jid, plan: plan, **kwargs)
 57end
 58
 59CONFIG = {
 60	sgx: "sgx",
 61	component: {
 62		jid: "component"
 63	},
 64	creds: {
 65		account: "test_bw_account",
 66		username: "test_bw_user",
 67		password: "test_bw_password"
 68	},
 69	notify_from: "notify_from@example.org",
 70	activation_amount: 1,
 71	activation_amount_accept: 1,
 72	plans: [
 73		{
 74			name: "test_usd",
 75			currency: :USD,
 76			monthly_price: 10000,
 77			messages: :unlimited,
 78			minutes: { included: 10440, price: 87 }
 79		},
 80		{
 81			name: "test_bad_currency",
 82			currency: :BAD
 83		},
 84		{
 85			name: "test_cad",
 86			currency: :CAD,
 87			monthly_price: 10000
 88		}
 89	],
 90	braintree: {
 91		merchant_accounts: {
 92			USD: "merchant_usd"
 93		}
 94	},
 95	sip: {
 96		realm: "sip.example.com",
 97		app: "sipappid"
 98	},
 99	credit_card_url: ->(*) { "http://creditcard.example.com" },
100	electrum_notify_url: ->(*) { "http://notify.example.com" },
101	keep_area_codes: ["556"],
102	keep_area_codes_in: { account: "moveto", site: "movetosite" },
103	upstream_domain: "example.net",
104	approved_domains: {
105		"approved.example.com": nil,
106		"refer.example.com": "refer_to"
107	}
108}.freeze
109
110def panic(e)
111	raise e
112end
113
114LOG = Class.new {
115	def child(*)
116		Minitest::Mock.new
117	end
118}.new.freeze
119
120BLATHER = Class.new {
121	def <<(*); end
122}.new.freeze
123
124def execute_command(
125	iq=Blather::Stanza::Iq::Command.new.tap { |i| i.from = "test@example.com" },
126	blather: BLATHER,
127	&blk
128)
129	Command::Execution.new(
130		Minitest::Mock.new,
131		blather,
132		:to_s.to_proc,
133		iq
134	).execute(&blk).sync
135end
136
137class Matching
138	def initialize(&block)
139		@block = block
140	end
141
142	def ===(other)
143		@block.call(other)
144	end
145end
146
147class PromiseMock < Minitest::Mock
148	def then(succ=nil, _=nil)
149		if succ
150			succ.call(self)
151		else
152			yield self
153		end
154	end
155end
156
157class FakeTelSelections
158	def initialize
159		@selections = {}
160	end
161
162	def set(jid, tel)
163		@selections[jid] = EMPromise.resolve(TelSelections::HaveTel.new(tel))
164	end
165
166	def delete(jid)
167		@selections.delete(jid)
168		EMPromise.resolve("OK")
169	end
170
171	def [](jid)
172		@selections.fetch(jid) do
173			TelSelections::ChooseTel.new
174		end
175	end
176end
177
178class FakeRedis
179	def initialize(values={})
180		@values = values
181	end
182
183	def set(key, value)
184		@values[key] = value
185		EMPromise.resolve("OK")
186	end
187
188	def setex(key, _expiry, value)
189		set(key, value)
190	end
191
192	def mget(*keys)
193		EMPromise.all(keys.map(&method(:get)))
194	end
195
196	def get(key)
197		EMPromise.resolve(@values[key])
198	end
199
200	def getbit(key, bit)
201		get(key).then { |v| v.to_i.to_s(2)[bit].to_i }
202	end
203
204	def hget(key, field)
205		@values.dig(key, field)
206	end
207
208	def hincrby(key, field, incrby)
209		@values[key] ||= {}
210		@values[key][field] ||= 0
211		@values[key][field] += incrby
212	end
213
214	def sadd(key, member)
215		@values[key] ||= Set.new
216		@values[key] << member
217	end
218
219	def srem(key, member)
220		@values[key].delete(member)
221	end
222
223	def scard(key)
224		@values[key]&.size || 0
225	end
226
227	def expire(_, _); end
228
229	def exists(*keys)
230		EMPromise.resolve(
231			@values.select { |k, _| keys.include? k }.size
232		)
233	end
234
235	def lindex(key, index)
236		get(key).then { |v| v&.fetch(index) }
237	end
238end
239
240class FakeDB
241	class MultiResult
242		def initialize(*args)
243			@results = args
244		end
245
246		def to_a
247			@results.shift
248		end
249	end
250
251	def initialize(items={})
252		@items = items
253	end
254
255	def query_defer(_, args)
256		EMPromise.resolve(@items.fetch(args, []).to_a)
257	end
258
259	def query_one(_, *args, field_names_as: :symbol, default: nil)
260		row = @items.fetch(args, []).to_a.first
261		row = row.transform_keys(&:to_sym) if row && field_names_as == :symbol
262		EMPromise.resolve(row || default)
263	end
264end
265
266class FakeLog
267	def initialize
268		@logs = []
269	end
270
271	def respond_to_missing?(*)
272		true
273	end
274
275	def method_missing(*args)
276		@logs << args
277	end
278end
279
280class FakeIBRRepo
281	def initialize(registrations={})
282		@registrations = registrations
283	end
284
285	def registered?(jid, from:)
286		@registrations.dig(jid.to_s, from.to_s) || false
287	end
288end
289
290module EventMachine
291	class << self
292		# Patch EM.add_timer to be instant in tests
293		alias old_add_timer add_timer
294		def add_timer(*args, &block)
295			args[0] = 0
296			old_add_timer(*args, &block)
297		end
298	end
299end
300
301module Minitest
302	class Test
303		def self.property(m, &block)
304			define_method("test_#{m}") do
305				property_of(&block).check { |args| send(m, *args) }
306			end
307		end
308
309		def self.em(m)
310			alias_method "raw_#{m}", m
311			define_method(m) do
312				EM.run do
313					Fiber.new {
314						begin
315							send("raw_#{m}")
316						ensure
317							EM.stop
318						end
319					}.resume
320				end
321			end
322		end
323	end
324end