hyperoceanic

hyperoceanic

Testing ECSx.Components and Systems

I"m working on a Kanban simulator, and I’m having some problems getting started with testing. The error message I’m getting is

** (ArgumentError) errors were found at the given arguments:

 * 1st argument: the table identifier refers to an ETS table with insufficient access rights
    

Here’s my Component:

defmodule Taiichi.Components.StoryPoints do
  use ECSx.Component,
    value: :integer
end

And here’s my test:

 test "sets remaining points to story points if not assigned" do

        task = Ecto.UUID.generate()
        StoryPoints.add(task, 100)
        
        found = StoryPoints.get_all()
        assert Enum.any?(found)
    end

It’s the StoryPoints.add(task, 100) that causes the error.

I had a look at the tests in the main Ecsx repo and in the example Ship repo but could not see any tests that did something similar with a Component in an application vs one that is stood up in the test itself.

I can insert the component if I do it in the Manager, back in the main application.

I think the issue is that the ETS table is owned by the application process not the test process but I could not find an example that showed how to get this to work.

Is there some guidance on using ETS-dependent code in tests?

First Post!

APB9785

APB9785

Creator of ECSx

Hi Mark, and thanks for using ECSx!

You’re right that ETS tables are owned by the Manager and cannot be written to from any other process. This is to ensure data consistency while Systems are running. However, there is of course a need for various client processes to make updates to Component data, so the answer in ECSx is ClientEvents.

In your case, the client is the Test process, which would like to create a StoryPoints Component. Instead of having that Test process attempt StoryPoints.add/2 directly, it should call ECSx.ClientEvents.add/2 to create a ClientEvent. Then a ClientEventHandler System (which runs from the Manager) can detect the event and add the StoryPoint Component.

This strategy for updating Components via ClientEvents is also used in the Ship tutorial - Client Input via LiveView

Some other notes specifically about testing:

ECSx (and Entity-Component-System apps in general) manage data over time, using tick intervals. This means that updates to data don’t happen right away, but wait until the next tick. This can cause some difficulties when testing, especially if the tick rate is low. You can either raise the tick_rate config in testing, or manually send a :tick message to the Manager when needed.

Also, since ECSx uses global shared state (GenServer, ETS), the effects of your updates to Component data can be seen by all other tests. There is no “sandboxing” of the Component data to isolate side-effects to an individual test. If you create different entity IDs for each test, you can somewhat avoid conflicts, but patterns such as

found = StoryPoints.get_all()
assert Enum.any?(found)

can be problematic if another test is also creating StoryPoints components (each test will see the components created by the other test).

Where Next?

Popular in Questions Top

siddhant3030
Hi, I have to write a raw query for one of my project. But till now I have used ecto queries and don’t have much experience writing raw ...
New
Darmani72
If I have a post route which an argument: post /my_post_route/:my_param1, MyController.my_post_handler How would get the post params ...
New
Fl4m3Ph03n1x
About me? ( if you have nothing better to do than reading about some random guy in the internet :stuck_out_tongue: ) Hello all, this is ...
New
New
pmjoe
I have a relationship of love and hate with Elixir. Lots of things are just absolutely right, but there are some things that are kind of ...
New
vonH
When I run the Plug and I recompile I wind up having to use Ctrl C to quit iex and start again. Witht the help of rlwrap I can use the cu...
New
Lily
In templates/appointment/index.html.eex: <%= for appointment <- @appointments do %> <tr> <td><%= appoi...
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
bsollish-terakeet
Credo is smart enough to check for (something like) this: assert length(the_list) == 0 with this response: Checking if an enum is empt...
New
sergio_101
I am VERY much an elixir newbie. I have taken one elixir course and one phoenix course on Udemy. During that course, I saw the instructor...
New

Other popular topics Top

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
electic
Hi, I am new to Elixir. I am trying to use the DateTime component to insert a date into MySQL however the there seems to be no way to fo...
New
jononomo
I am trying to figure out how Mix knows whether the environment is test, dev, or prod -- where is this set? Thanks.
New
alice
Hey, Just curious what are the main benefits of Elixir compared to Clojure? When is Elixir more useful than Clojure and vice versa? Th...
New
jerry
Good day to you all. I have been struggling to get a query involving like and ilike to work. Can anyone assist me on this, please? pro...
New
chrismccord
This release brings a number of exciting features, including integration with the new Phoenix LiveDashboard and Phoenix LiveView. There h...
New
Qqwy
Original source of discussion: This topic on the Pragmatic Programmers' Functional Web Development with Elixir, OTP, and Phoenix forum. ...
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 31107 143
New
romenigld
I am trying to run a deploy with docker and I successfully runned with this command: docker build -t romenigld/blog-prod . but when I t...
New
svb
Hi! Currently I want to submit a form by pressing the Enter key. However, since my input field is of type “textarea” this is just adds a...
New

We're in Beta

About us Mission Statement