Maxximiliann

Maxximiliann

* (CompileError): undefined function

defmodule Benchmark do

	def main do
		module_function = Foo.main
		_ = module_function
	end

	def run(params) do
		number_of_trials = Enum.at(params, 0)
		total_warm_up_cycles = Enum.at(params, 1)

		IO.puts("Warming up . . . initiating first trial of #{number_of_trials}.")

		start_time = System.monotonic_time(:millisecond)

		main = main()
		_ = main

		end_time = System.monotonic_time(:millisecond)
		elapsed_time = (end_time - start_time) / 1000

		results = %{time: [elapsed_time], acc: 1}  

		run(results, number_of_trials, totaL_warm_up_cycles)
	end


	def run(results, number_of_trials, total_warm_up_cycles) do
		start_time = System.monotonic_time(:millisecond)

		main = main()
		_ = main

		end_time = System.monotonic_time(:millisecond)
		new_time = (end_time - start_time) / 1000 
		updated_acc = results.acc + 1

		average_time = 
		cond do 
			results.acc == total_warm_up_cycles -> new_time
			results.acc > total_warm_up_cycles -> Enum.sum(results.time) / (updated_acc - (total_warm_up_cycles + 1))
			results.acc < total_warm_up_cycles -> []
		end

		results = 
		if results.acc >= total_warm_up_cycles do
			Enum.map(results, fn _map ->
				%{
					time: List.insert_at(results.time, 0, new_time),
					acc: updated_acc,
					average_time: average_time
				}
			end)
			|> List.first
		else
			Enum.map(results, fn _map ->
				%{
					time: [],
					acc: updated_acc,
					average_time: 0
				}
			end)
			|> List.first
		end

		cond do 
			results.acc < (total_warm_up_cycles + 1) -> IO.puts("Warming up . . . Trial number #{results.acc} of #{number_of_trials}.")	
			run(module_function, results, number_of_trials, total_warm_up_cycles)

			results.acc == (total_warm_up_cycles + 1) -> IO.puts("Average execution time after #{results.acc} trials of #{number_of_trials}: #{Float.floor(results.average_time, 3)} seconds.")
			run(module_function, results, number_of_trials, total_warm_up_cycles)

			results.acc < number_of_trials -> IO.puts("Average execution time after #{results.acc} trials of #{number_of_trials}: #{Float.floor(results.average_time, 3)} seconds.")
			run(module_function, results, number_of_trials, total_warm_up_cycles)

			results.acc == number_of_trials -> IO.puts("Final trial - Average execution time: #{Float.floor(results.average_time, 3)} seconds.")
		end
	end		
	
end

This is designed to accept the following command:
iex(1)> Benchmark.run([20, 5])

Which would then output:

Warming up . . . initiating first trial of 20.
Warming up . . . Trial number 2 of 20.
Warming up . . . Trial number 3 of 20.
Warming up . . . Trial number 4 of 20.
Warming up . . . Trial number 5 of 20.
Average execution time after 6 trials of 20: 0.754 seconds.
Average execution time after 7 trials of 20: 0.754 seconds.
Average execution time after 8 trials of 20: 0.549 seconds.
Average execution time after 9 trials of 20: 0.48 seconds.
Average execution time after 10 trials of 20: 0.448 seconds.
Average execution time after 11 trials of 20: 0.496 seconds.
Average execution time after 12 trials of 20: 0.471 seconds.
Average execution time after 13 trials of 20: 0.499 seconds.
Average execution time after 14 trials of 20: 0.485 seconds.
Average execution time after 15 trials of 20: 0.47 seconds.
Average execution time after 16 trials of 20: 0.457 seconds.
Average execution time after 17 trials of 20: 0.447 seconds.
Average execution time after 18 trials of 20: 0.438 seconds.
Average execution time after 19 trials of 20: 0.431 seconds.
Final trial - Average execution time: 0.427 seconds.
:ok

When attempting to compile this module, however, it triggers the following error:

== Compilation error in file lib/Benchmark.ex ==
** (CompileError) lib/Benchmark.ex:24: undefined function totaL_warm_up_cycles/0

It looks like total_warm_up_cycles = Enum.at(params, 1) is being ignored? If so, why and how is this resolved?

Ideally I’d love to get this to run with something like iex(2)> Benchmark.run([MODULE.FUNCTION, 20, 5]) which would make this more dynamic since the particular module function to be benchmarked wouldn’t have to be hardcoded. After spending dozens of hours searching far and wide, though, I can’t find any resources that cover how exactly this is done. Is this even possible?

Thanks for all your help and opinions! :slight_smile:

Marked As Solved

Nicd

Nicd

totaL_warm_up_cycles has a big L, it’s a typo.

Also Liked

NobbZ

NobbZ

There is an uppercase L in the usage of the variable, but not in it’s definition

kokolegorille

kokolegorille

You put a capital L in the second totaL_warm_up_cycles.

Nicd

Nicd

As for this part, have you tried for example Benchmark.run(&Module.function/n, 20, 5) (substituting module, function, and n for the respective module, function, and arity)? Then you could call the function with function.(args) (note the period).

EDIT: For more information about function capturing, see: https://elixir-lang.org/getting-started/modules-and-functions.html#function-capturing

Where Next?

Popular in Questions Top

_russellb
I want to try my hand at web scraping. What tools/libraries do I need to use. I’m hoping to turn this into something professional so don’...
New
chrisalley
ExUnit now has describe blocks which is a welcome addition coming from RSpec. In the docs, it states that nested hierarchies of describe ...
New
tduccuong
Hi, is there any work on GUI with Elixir, that is similar to Electron/Javascript? My idea is to bundle Phoenix and BEAM into a single se...
New
ycv005
I have followed this StackOverflow post to install the specific version of Erlang. And When I am running mix ecto.setup then getting fol...
New
freewebwithme
Using vs code and installed ElixirLS: support and debugger. And I got an error popped up on start up says Failed to run ‘elixir’ comma...
New
chensan
I have a User schema with a :from_id field set to type :string: defmodule TweetBot.Repo.Migrations.CreateUsers do use Ecto.Migration ...
New
nsuchy
Hi. I’ve noticed that Windows Powershell has it’s own IEX command and you cannot access Elixir’s IEX due to the conflict. This isn’t a cr...
New
dotdotdotPaul
Okay, I’m having a heck of a time trying to figure out how to best handle the validation of belongs_to associations in Ecto. I’m sure I’...
New
openscript
Hello! Sorry for this astonishing simple question, but I’m really stuck. I try to set up the intellij-elixir plugin, but I don’t know ho...
New
hariharasudhan94
Lets say I have map like this fetching from my database %{"_id" =&gt; #BSON.ObjectId&lt;58eb1a7a9ad169198c3dXXXX&gt;, "email" =&gt; ...
New

Other popular topics Top

sen
Hi All, I set a environment variables in dev.exs , like below code. when i start server, how can i set the ${enable} value? thanks. d...
New
TunkShif
This post is an instruction guide to help you setup your Neovim for Elixir development from scratch. It includes general information on h...
274 41539 114
New
skosch
To my knowledge, put_in, Map.update etc. all have the one limitation of not automatically creating intermediate keys when needed (for exa...
New
greenz1
I have a phoenix application from which a user can download multiple(5-6) files of size 1MB. I couldn’t find anything related to sending ...
New
msaraiva
Surface is an experimental library built on top of Phoenix LiveView and its new LiveComponent API that aims to provide a more declarative...
564 43622 214
New
SoCreat
i’m a new one to elixir which editor can i use vs code? or atom? Thanks! :smiley:
New
AstonJ
Please see the new poll here: Which code editor or IDE do you use? (Poll) (2022 Edition) It’s been a while since we first asked this, I...
208 31142 143
New
ashish173
I am using Ecto timestamps with postgres, I can see the timestamps() use the :naive_dateime but for my use case I wanted to store the ti...
New
klo
Got a question about when to concat vs. prepending items to list then reversing to achieve appending. So i know lists boil down to [1 | ...
New
Brian
What is the proper way to load a module from a file in to IEX? In the python world, doing something like this pretty standard: from ....
New

We're in Beta

About us Mission Statement