Skip to content

Commit

Permalink
[IMP] snippets: move all work from parent to mp workers
Browse files Browse the repository at this point in the history
In `convert_html_columns()`, we select 100MiB worth of DB tuples and pass them
to a ProcessPoolExecutor together with a converter callable. So far, the
converter returns all tuples, changed or unchanged together with the
information if it has changed something. All this is returned through IPC to
the parent process. In the parent process, the caller only acts on the changed
tuples, though, the rest is ignored. In any scenario I've seen, only a small
proportion of the input tuples is actually changed, meaning that a large
proportion is returned through IPC unnecessarily.

What makes it worse is that processing of the converted results in the parent
process is often slower than the conversion, leading to two effects:
1) The results of all workers sit in the parent process's memory, possibly
   leading to MemoryError (upg-2021031)
2) The parallel processing is being serialized on the feedback, defeating a
   large part of the intended performance gains

To improve this, this commit
- moves all work into the workers, meaning not just the conversion filter, but
  also the DB query as well as the DB updates.
  - by doing so reduces the amount of data passed by IPC to just the query
    texts
  - by doing so distributes the data held in memory to all worker processes
- reduces the chunk size by one order of magnitude, which means
  - a lot less memory used at a time
  - a lot better distribution of "to-be-changed" rows when these rows are
    clustered in the table

All in all, in my test case, this
- reduces maximum process size in memory to 300MiB for all processes compared
  to formerly >2GiB (and MemoryError) in the parent process
- reduces runtime from 17 minutes to less than 2 minutes
  • Loading branch information
cawo-odoo committed Sep 16, 2024
1 parent a84c23a commit 2c980fa
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 29 deletions.
10 changes: 7 additions & 3 deletions src/base/tests/test_util.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
except ImportError:
import mock

from odoo import SUPERUSER_ID, api
from odoo.osv.expression import FALSE_LEAF, TRUE_LEAF
from odoo.tools import mute_logger
from odoo.tools.safe_eval import safe_eval
Expand Down Expand Up @@ -1436,6 +1437,8 @@ def not_doing_anything_converter(el):

class TestHTMLFormat(UnitTestCase):
def testsnip(self):
cr = self.registry.cursor()
env = api.Environment(cr, SUPERUSER_ID, {})
view_arch = """
<html>
<div class="fake_class_not_doing_anything"><br/></div>
Expand All @@ -1445,7 +1448,7 @@ def testsnip(self):
</script>
</html>
"""
view_id = self.env["ir.ui.view"].create(
view_id = env["ir.ui.view"].create(
{
"name": "not_for_anything",
"type": "qweb",
Expand All @@ -1454,15 +1457,16 @@ def testsnip(self):
"arch_db": view_arch,
}
)
cr = self.env.cr
snippets.convert_html_content(
cr,
snippets.html_converter(
not_doing_anything_converter, selector="//*[hasclass('fake_class_not_doing_anything')]"
),
)
util.invalidate(view_id)
res = self.env["ir.ui.view"].search_read([("id", "=", view_id.id)], ["arch_db"])
res = env["ir.ui.view"].search_read([("id", "=", view_id.id)], ["arch_db"])
cr.commit()
cr.close()
self.assertEqual(len(res), 1)
oneline = lambda s: re.sub(r"\s+", " ", s.strip())
self.assertEqual(oneline(res[0]["arch_db"]), oneline(view_arch))
Expand Down
74 changes: 48 additions & 26 deletions src/util/snippets.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
# -*- coding: utf-8 -*-
import concurrent
import inspect
import logging
import re
Expand All @@ -11,6 +12,11 @@
from psycopg2.extensions import quote_ident
from psycopg2.extras import Json

try:
from odoo.sql_db import db_connect
except ImportError:
from openerp.sql_db import db_connect

from .const import NEARLYWARN
from .exceptions import MigrationError
from .helpers import table_of_model
Expand Down Expand Up @@ -243,31 +249,39 @@ def _dumps(self, node):


class Convertor:
def __init__(self, converters, callback):
def __init__(self, converters, callback, dbname, update_query):
self.converters = converters
self.callback = callback
self.dbname = dbname
self.update_query = update_query

def __call__(self, row):
def __call__(self, query):
converters = self.converters
columns = self.converters.keys()
converter_callback = self.callback
res_id, *contents = row
changes = {}
for column, content in zip(columns, contents):
if content and converters[column]:
# jsonb column; convert all keys
new_content = {}
has_changed, new_content["en_US"] = converter_callback(content.pop("en_US"))
if has_changed:
for lang, value in content.items():
_, new_content[lang] = converter_callback(value)
new_content = Json(new_content)
else:
has_changed, new_content = converter_callback(content)
changes[column] = new_content
if has_changed:
changes["id"] = res_id
return changes if "id" in changes else None
update_query = self.update_query
with db_connect(self.dbname).cursor() as cr:
cr.execute(query)
for row in cr.fetchall():
res_id, *contents = row
changes = {}
for column, content in zip(columns, contents):
if content and converters[column]:
# jsonb column; convert all keys
new_content = {}
has_changed, new_content["en_US"] = converter_callback(content.pop("en_US"))
if has_changed:
for lang, value in content.items():
_, new_content[lang] = converter_callback(value)
new_content = Json(new_content)
else:
has_changed, new_content = converter_callback(content)
changes[column] = new_content
if has_changed:
changes["id"] = res_id
if "id" in changes:
cr.execute(update_query, changes)
cr.commit()


def convert_html_columns(cr, table, columns, converter_callback, where_column="IS NOT NULL", extra_where="true"):
Expand Down Expand Up @@ -305,17 +319,25 @@ def convert_html_columns(cr, table, columns, converter_callback, where_column="I
update_sql = ", ".join(f'"{column}" = %({column})s' for column in columns)
update_query = f"UPDATE {table} SET {update_sql} WHERE id = %(id)s"

cr.commit()
with ProcessPoolExecutor(max_workers=get_max_workers()) as executor:
convert = Convertor(converters, converter_callback)
for query in log_progress(split_queries, logger=_logger, qualifier=f"{table} updates"):
cr.execute(query)
for data in executor.map(convert, cr.fetchall(), chunksize=1000):
if data:
cr.execute(update_query, data)
convert = Convertor(converters, converter_callback, cr.dbname, update_query)
futures = [executor.submit(convert, query) for query in split_queries]
for future in log_progress(
concurrent.futures.as_completed(futures),
logger=_logger,
qualifier=f"{table} updates",
size=len(split_queries),
estimate=False,
log_hundred_percent=True,
):
# just for raising any worker exception
future.result()
cr.commit()


def determine_chunk_limit_ids(cr, table, column_arr, where):
bytes_per_chunk = 100 * 1024 * 1024
bytes_per_chunk = 10 * 1024 * 1024
columns = ", ".join(quote_ident(column, cr._cnx) for column in column_arr if column != "id")
cr.execute(
f"""
Expand Down

0 comments on commit 2c980fa

Please sign in to comment.