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	Customer.extract(
 51		customer_id,
 52		jid,
 53		plan_name: plan_name,
 54		expires_at: expires_at,
 55		auto_top_up_amount: auto_top_up_amount,
 56		**kwargs
 57	)
 58end
 59
 60CONFIG = {
 61	sgx: "sgx",
 62	component: {
 63		jid: "component"
 64	},
 65	creds: {
 66		account: "test_bw_account",
 67		username: "test_bw_user",
 68		password: "test_bw_password"
 69	},
 70	notify_from: "notify_from@example.org",
 71	activation_amount: 1,
 72	activation_amount_accept: 1,
 73	plans: [
 74		{
 75			name: "test_usd",
 76			currency: :USD,
 77			monthly_price: 10000,
 78			messages: :unlimited,
 79			minutes: { included: 10440, price: 87 }
 80		},
 81		{
 82			name: "test_bad_currency",
 83			currency: :BAD
 84		},
 85		{
 86			name: "test_cad",
 87			currency: :CAD,
 88			monthly_price: 10000
 89		}
 90	],
 91	braintree: {
 92		merchant_accounts: {
 93			USD: "merchant_usd"
 94		}
 95	},
 96	sip: {
 97		realm: "sip.example.com",
 98		app: "sipappid"
 99	},
100	credit_card_url: ->(*) { "http://creditcard.example.com" },
101	electrum_notify_url: ->(*) { "http://notify.example.com" },
102	keep_area_codes: ["556"],
103	keep_area_codes_in: {
104		account: "moveto",
105		site_id: "movetosite",
106		sip_peer_id: "movetopeer"
107	},
108	upstream_domain: "example.net",
109	approved_domains: {
110		"approved.example.com": nil,
111		"refer.example.com": "refer_to"
112	}
113}.freeze
114
115def panic(e)
116	raise e
117end
118
119LOG = Class.new {
120	def child(*)
121		Minitest::Mock.new
122	end
123
124	def info(*); end
125}.new.freeze
126
127def log
128	LOG
129end
130
131BLATHER = Class.new {
132	def <<(*); end
133}.new.freeze
134
135def execute_command(
136	iq=Blather::Stanza::Iq::Command.new.tap { |i| i.from = "test@example.com" },
137	blather: BLATHER,
138	&blk
139)
140	Command::Execution.new(
141		Minitest::Mock.new,
142		blather,
143		:to_s.to_proc,
144		iq
145	).execute(&blk).sync
146end
147
148class Matching
149	def initialize(&block)
150		@block = block
151	end
152
153	def ===(other)
154		@block.call(other)
155	end
156end
157
158class PromiseMock < Minitest::Mock
159	def then(succ=nil, _=nil)
160		if succ
161			succ.call(self)
162		else
163			yield self
164		end
165	end
166end
167
168class FakeTelSelections
169	def initialize
170		@selections = {}
171	end
172
173	def set(jid, tel)
174		@selections[jid] = EMPromise.resolve(TelSelections::HaveTel.new(tel))
175	end
176
177	def delete(jid)
178		@selections.delete(jid)
179		EMPromise.resolve("OK")
180	end
181
182	def [](jid)
183		@selections.fetch(jid) do
184			TelSelections::ChooseTel.new
185		end
186	end
187end
188
189class FakeRedis
190	def initialize(values={})
191		@values = values
192	end
193
194	def set(key, value)
195		@values[key] = value
196		EMPromise.resolve("OK")
197	end
198
199	def setex(key, _expiry, value)
200		set(key, value)
201	end
202
203	def mget(*keys)
204		EMPromise.all(keys.map(&method(:get)))
205	end
206
207	def get(key)
208		EMPromise.resolve(@values[key])
209	end
210
211	def getbit(key, bit)
212		get(key).then { |v| v.to_i.to_s(2)[bit].to_i }
213	end
214
215	def bitfield(key, *ops)
216		get(key).then do |v|
217			bits = v.to_i.to_s(2)
218			ops.each_slice(3).map do |(op, encoding, offset)|
219				raise "unsupported bitfield op" unless op == "GET"
220				raise "unsupported bitfield op" unless encoding == "u1"
221
222				bits[offset].to_i
223			end
224		end
225	end
226
227	def hget(key, field)
228		@values.dig(key, field)
229	end
230
231	def hincrby(key, field, incrby)
232		@values[key] ||= {}
233		@values[key][field] ||= 0
234		@values[key][field] += incrby
235	end
236
237	def sadd(key, member)
238		@values[key] ||= Set.new
239		@values[key] << member
240	end
241
242	def srem(key, member)
243		@values[key].delete(member)
244	end
245
246	def scard(key)
247		@values[key]&.size || 0
248	end
249
250	def expire(_, _); end
251
252	def exists(*keys)
253		EMPromise.resolve(
254			@values.select { |k, _| keys.include? k }.size
255		)
256	end
257
258	def lindex(key, index)
259		get(key).then { |v| v&.fetch(index) }
260	end
261end
262
263class FakeDB
264	class MultiResult
265		def initialize(*args)
266			@results = args
267		end
268
269		def to_a
270			@results.shift
271		end
272	end
273
274	def initialize(items={})
275		@items = items
276	end
277
278	def query_defer(_, args)
279		EMPromise.resolve(@items.fetch(args, []).to_a)
280	end
281
282	def query_one(_, *args, field_names_as: :symbol, default: nil)
283		row = @items.fetch(args, []).to_a.first
284		row = row.transform_keys(&:to_sym) if row && field_names_as == :symbol
285		EMPromise.resolve(row || default)
286	end
287end
288
289class FakeLog
290	def initialize
291		@logs = []
292	end
293
294	def respond_to_missing?(*)
295		true
296	end
297
298	def method_missing(*args)
299		@logs << args
300	end
301end
302
303class FakeIBRRepo
304	def initialize(registrations={})
305		@registrations = registrations
306	end
307
308	def registered?(jid, from:)
309		@registrations.dig(jid.to_s, from.to_s) || false
310	end
311end
312
313module EventMachine
314	class << self
315		# Patch EM.add_timer to be instant in tests
316		alias old_add_timer add_timer
317		def add_timer(*args, &block)
318			args[0] = 0
319			old_add_timer(*args, &block)
320		end
321	end
322end
323
324module Minitest
325	class Test
326		def self.property(m, &block)
327			define_method("test_#{m}") do
328				property_of(&block).check { |args| send(m, *args) }
329			end
330		end
331
332		def self.em(m)
333			alias_method "raw_#{m}", m
334			define_method(m) do
335				EM.run do
336					Fiber.new {
337						begin
338							send("raw_#{m}")
339						ensure
340							EM.stop
341						end
342					}.resume
343				end
344			end
345		end
346	end
347end