It's been some time now with ArangoDB. Development with it along with Python is very straight forward and quick.
However, I am recently facing some issues with the production side, which is a bare metal with these specs:
The ArangoDB versions are:
LOCAL:
PRODUCTION:
The issue is the long time of quite simple queries which in local are a matter of milliseconds, while in production a matter of whole seconds. I also should point out that the two collections used in the query are bigger in production (searches has 41k docs, results has 350k docs) than in local (searches has 1k docs, results has 27k docs), however I did notice a slow habit in production even with smaller numbers.
Here's the production profiled query:
Query String (415 chars, cacheable: false):
FOR entry in searches
FILTER entry.session_key == '21150583'
LET counters = (
FOR result in results_linkers
FILTER result.search_key == entry._key AND result.validation_automatic.status == 'completed'
COLLECT action = result.validation_automatic.action WITH COUNT INTO action_counter
RETURN { 'action': action, 'count': action_counter }
)
LIMIT 20
RETURN { 'search': entry, 'counters': counters }
Execution plan:
Id NodeType Calls Items Runtime [s] Comment
1 SingletonNode 1 1 0.00000 * ROOT
2 EnumerateCollectionNode 1 20 0.00737 - FOR entry IN searches /* full collection scan */ FILTER (entry.`session_key` == "21150583") /* early pruning */
14 LimitNode 1 20 0.00001 - LIMIT 0, 20
18 SubqueryStartNode 1 40 0.00001 - LET counters = ( /* subquery begin */
6 EnumerateCollectionNode 7136 7135580 3.33341 - FOR result IN results_linkers /* full collection scan, projections: `search_key`, `validation_automatic` */
7 CalculationNode 7136 7135580 1.36629 - LET #9 = ((result.`search_key` == entry.`_key`) && (result.`validation_automatic`.`status` == "completed")) /* simple expression */ /* collections used: result : results_linkers, entry : searches */
8 FilterNode 2 1854 0.12298 - FILTER #9
9 CalculationNode 2 1854 0.00024 - LET #11 = result.`validation_automatic`.`action` /* attribute expression */ /* collections used: result : results_linkers */
10 CollectNode 1 56 0.00013 - COLLECT action = #11 AGGREGATE action_counter = LENGTH() /* hash */
17 SortNode 1 56 0.00001 - SORT action ASC /* sorting strategy: standard */
11 CalculationNode 1 56 0.00002 - LET #13 = { "action" : action, "count" : action_counter } /* simple expression */
19 SubqueryEndNode 1 20 0.00001 - RETURN #13 ) /* subquery end */
15 CalculationNode 1 20 0.00001 - LET #15 = { "search" : entry, "counters" : counters } /* simple expression */ /* collections used: entry : searches */
16 ReturnNode 1 20 0.00000 - RETURN #15
Indexes used:
none
Optimization rules applied:
Id RuleName
1 move-calculations-up
2 move-filters-up
3 move-calculations-up-2
4 move-filters-up-2
5 move-calculations-down
6 reduce-extraction-to-projection
7 move-filters-into-enumerate
8 splice-subqueries
Query Statistics:
Writes Exec Writes Ign Scan Full Scan Index Filtered Peak Mem [b] Exec Time [s]
0 0 7176820 0 7174966 655360 4.83091
Query Profile:
Query Stage Duration [s]
initializing 0.00000
parsing 0.00004
optimizing ast 0.00001
loading collections 0.00000
instantiating plan 0.00003
optimizing plan 0.00031
executing 4.83051
finalizing 0.00002
Here's the local one:
Query String (414 chars, cacheable: false):
FOR entry in searches
FILTER entry.session_key == '3307542'
LET counters = (
FOR result in results_linkers
FILTER result.search_key == entry._key AND result.validation_automatic.status == 'completed'
COLLECT action = result.validation_automatic.action WITH COUNT INTO action_counter
RETURN { 'action': action, 'count': action_counter }
)
LIMIT 20
RETURN { 'search': entry, 'counters': counters }
Execution plan:
Id NodeType Calls Items Runtime [s] Comment
1 SingletonNode 1 1 0.00001 * ROOT
2 EnumerateCollectionNode 1 1 0.00097 - FOR entry IN searches /* full collection scan */ FILTER (entry.`session_key` == "3307542") /* early pruning */
14 LimitNode 1 1 0.00001 - LIMIT 0, 20
18 SubqueryStartNode 1 2 0.00001 - LET counters = ( /* subquery begin */
6 EnumerateCollectionNode 28 27775 0.02148 - FOR result IN results_linkers /* full collection scan, projections: `search_key`, `validation_automatic` */
7 CalculationNode 28 27775 0.00866 - LET #9 = ((result.`search_key` == entry.`_key`) && (result.`validation_automatic`.`status` == "completed")) /* simple expression */ /* collections used: result : results_linkers, entry : searches */
8 FilterNode 1 97 0.00169 - FILTER #9
9 CalculationNode 1 97 0.00002 - LET #11 = result.`validation_automatic`.`action` /* attribute expression */ /* collections used: result : results_linkers */
10 CollectNode 1 3 0.00002 - COLLECT action = #11 AGGREGATE action_counter = LENGTH() /* hash */
17 SortNode 1 3 0.00001 - SORT action ASC /* sorting strategy: standard */
11 CalculationNode 1 3 0.00001 - LET #13 = { "action" : action, "count" : action_counter } /* simple expression */
19 SubqueryEndNode 1 1 0.00001 - RETURN #13 ) /* subquery end */
15 CalculationNode 1 1 0.00001 - LET #15 = { "search" : entry, "counters" : counters } /* simple expression */ /* collections used: entry : searches */
16 ReturnNode 1 1 0.00001 - RETURN #15
Indexes used:
none
Optimization rules applied:
Id RuleName
1 move-calculations-up
2 move-filters-up
3 move-calculations-up-2
4 move-filters-up-2
5 move-calculations-down
6 reduce-extraction-to-projection
7 move-filters-into-enumerate
8 splice-subqueries
Query Statistics:
Writes Exec Writes Ign Scan Full Scan Index Filtered Peak Mem [b] Exec Time [s]
0 0 28901 0 28804 360448 0.03434
Query Profile:
Query Stage Duration [s]
initializing 0.00002
parsing 0.00012
optimizing ast 0.00002
loading collections 0.00001
instantiating plan 0.00007
optimizing plan 0.00117
executing 0.03293
finalizing 0.00003
Another detail to add up to the story is that I'm using the database with "single" big collections and retrieve the data looking up for the keys, like an old fashioned relational database. Is that also the wrong approach here? The collections are going to grow over the time so I'm a bit concerned on this.
Thanks for whoever will help me out on this!
I'd recommend you look at adding indexes to your collections.
The first index I would add is a Persistent index on the 'searches' collection, specifically on the key 'session_key'.
The second index I would add is a Persistent index on the 'results_linkers' collection, specifically on the key 'validation_automatic.status'. With this index, also test adding the second key to the index for the key 'result.validation_automatic.action', it's possible this will help in the collect command.
After adding the indexes, do the same query analysis and confirm your indexes are being picked up in the 'Indexes Used' section. If you aren't getting your indexes listed there, then they perform no advantage and you need to change the keys they index on to see if they trigger.
After adding an index you don't need to restart the server. When the next AQL query hits the server it will determine if any indexes help.
In my experience, indexes can improve query performance from 10x to 1000x depending on your query.
See if that helps.
When developing locally with a small data set, it's very easy to think your queries perform well only to find out in production they run slow.
The best way around this is to have a 'PVT - Production Verification Testing' space where you can test your new code on a database with a restore of production data, if you're able to do that.
See if these indexes help, and tune them if not, or ignore the index if it doesn't assist.