Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add workflows for local development using containerised YNR #2485

Draft
wants to merge 31 commits into
base: master
Choose a base branch
from

Conversation

jpluscplusm
Copy link
Collaborator

@jpluscplusm jpluscplusm commented Dec 30, 2024

This adds workflows, scripts, docs and in-repo infra for YNR to be invoked inside a container during local development. Non-local environments are not impacted by this change.

The branch being PR'd contains ~25 commits that generally build on top of each other in narrative fashion. These are presented as-is, as they contain information that's relevant about the development of the local-dev setup that results. They can, of course, be squashed if needed, but that loss of information should be an explicit project choice.


CI failure on tip of master previously blocking this PR

Edit: whilst this failure was previously reproducible, it has now fixed itself in CircleCI when the same commit is exercised (see the two runs visible at https://app.circleci.com/pipelines/github/DemocracyClub/yournextrepresentative/3940). I'll leave this section here for posterity, and in case the problem reoccurs. The first and last commits mentioned here have been removed.

Original: The first and last commits of this PR's stack relate to a single Python test that only started to fail after the stack was rebased on top of the recent #2481. The failure is present on the commit at the tip of the master branch, suggesting the failure is unrelated to this stack's changes. This pair of commits should be removed from this PR, pre-merge, after the root cause of the issue is addressed.


This stack is considered complete, but the PR is WIP. Changes may be needed to address various items in the PR checklist.

To review

  • Check you're happy with the rendered changes to README.md
  • Check out the PR's branch and ...
    • Follow and validate the instructions in docs/INSTALL.md
    • Follow and validate as many of the workflows in docs/DEVELOPMENT.md as you feel appropriate
    • Use the workflows in docs/DEVELOPMENT.md to test the process of making a "normal" code change to YNR.
  • Review container/build/Containerfile
  • Review scripts/*
  • Review docker-compose.yml
  • Review the changes to these pre-existing files:
    • ynr/settings/testing.py
    • requirements/base.txt
    • requirements/sopn_parsing.txt
    • ynr/apps/sopn_parsing/README.md

PR Checklist

  • Update changelog (ie https://keepachangelog.com/en/1.0.0/)
  • References a specific issue or if not, describes the bug or feature in detail
  • Note what card in Trello this work relates to
  • Instructions for how reviewers can test the code locally
  • Tests have been added and/or updated
  • Screenshot of the feature/bug fix (if applicable)
  • If any new text is added, it's internationalized
  • Any new elements have aria labels
  • No unintentional console.logs left behind after debugging
  • Did I use the clear and concise names for variables and functions?
  • Did I explain all possible solutions and why I chose the one I did?
  • Added any comments to make new functions clearer
  • Did I added or updated any new dependencies? Explain why.
  • Did I update the package.json and/or Pipfile.lock version if relevant?
  • Have I rebased with the latest version of master?
  • Added PR labels
  • Update any history/changelog file
  • Update any documentation
  • Update dev handbook

@jpluscplusm jpluscplusm requested a review from symroe December 30, 2024 11:36
Copy link
Member

@chris48s chris48s left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK then. Lets start off with the stuff that is just broken. I don't think these things necessarily are blockers before Sym can a go at bootstrapping the project, but @symroe - if you want to jump in and do it now, these are the things that I found were just broken and here are the workarounds/fixes. Hopefully that will stop you from wasting time hitting the same rough edges as I did.

docker-compose.yml Show resolved Hide resolved
docker-compose.yml Show resolved Hide resolved
## Assets #
###########
run date \
&& npm run build \
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

High priority (but harder)

I am getting the error Error compiling CSS package "all" with a clean container.

We talked about this one on the call. I don't have a proposed fix for this one, but I do have a clear set of steps I can give you to reproduce the problem:

  • Set DEBUG = True in your config file
  • Delete everything inside ynr/assets (other than .gitkeep)
  • Delete ynr/static
    (Deleting those things puts you into the state you're in when you do a clean checkout of the repo)
  • Build your image from scratch. I did this by podman image rm-ing my image, and then running podman compose build, but podman compose build --no-cache might do the job too.
  • podman compose up -d
  • Browse to http://127.0.0.1:8080/
  • You should get Error compiling CSS package "all"
    Screenshot at 2025-01-08 14-42-42
  • Note the ynr/assets is still empty
  • Now podman compose exec frontend bash in and run npm run build
  • Reload http://127.0.0.1:8080/
  • You should now see a styled page
  • Note ynr/assets now has files in it

I'm not sure off the top of my head what the answer is, but as I say I think it is to do with the fact that npm run build outputs files to ynr/assets but the whole of ynr is a bind mount. I haven't spent any more time on it. Let me know if it is useful to catch up on this one.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll take a look at this - thanks for the repro steps.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@chris48s I can reproduce this given the steps you've outlined.

I believe I'd already anticipated this chicken-and-egg problem, and prompted the user to solve it with steps 5 and 6 in docs/DEVELOPMENT.md#running-the-app.

Given the chicken-and-egg nature of the npm run build output being written to ynr, I'll have to defer to DC folks for their opinion on how reasonable it is to have this manual step as the solution. What's the cadence of change for the input to npm run build? Would it be more suitable to move the above steps 5 and 6 to sit as one-time, development preamble instructions in INSTALL.md ... and defer solving the DX challenge of a frontend dev making JS/CSS changes that require npm invocations until later?

Copy link
Member

@chris48s chris48s left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Medium priority

There's a bag of related issues around how we configure the application. I'm going to structure this as one "review" containing a series of inline comments, but I think its all kind of related. This might be one where its useful for us to collectively discuss the approach to application configuration (again) to hammer this out and work out wehat next. Maybe we can compare notes when we've both spun the app up locally.

This stuff does need sorting, but I don't think its a pre-requisite before Sym has a go at spinning up the application and the next actions are perhaps not 100% defined yet.

Comment on lines +47 to +51
`env/frontend.env`, like this:
```
DJANGO_SETTINGS_MODULE=ynr.settings.testing
```
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This settings file ynr.settings.testing is specifically for running tests, rather than for local development. For local dev, you want DJANGO_SETTINGS_MODULE=ynr.settings.base. base.py will also handle pulling in local.py if it exists.

Also, I think we should provide a env/frontend.env template with some sensible defaults in it in the repo, rather than do it as docs.
Bit of a tangent, but a relevant one: What are our thoughts on what goes in env vars and what goes in local settings? Are we envisaging that in local dev the only thing in env/frontend.env will be DJANGO_SETTINGS_MODULE? Is there another job of extracting settings into env vars ahead of us?

# Invoke a lightweight, post-build test that proves the container reaches a
# baseline level of correctness, whilst also generating .pyc files for faster
# app startup.
run python manage.py check --settings=ynr.settings.testing
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm. Does it make sense to run the system checks with the test settings? I'd be inclined to use ynr.settings.base here. I'm not sure if we have done it on YNR but a common use of system checks in django applications is making sure the app config is correct before launching the app.

Example from WDIV:
https://github.com/DemocracyClub/UK-Polling-Stations/blob/eebe8c724a3eb5bbba979937bca01b8482a34cc0/polling_stations/apps/data_finder/checks.py

@@ -29,7 +30,7 @@ def __getitem__(self, item):
RUNNING_TESTS = True

SECRET_KEY = "just here for testing"
ALLOWED_HOSTS = ["candidates.democracyclub.org.uk"]
ALLOWED_HOSTS = ["candidates.democracyclub.org.uk","*"]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What did we need to add this for?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I found that the setting as originally configured prevented the site from being accessible at localhost:8080 or similar, as the Host header (or perhaps lack of one) was rejected. I'm unsure if the check overlaps with a DEBUG setting that's more lenient, maybe, but at the point I needed to see the site on localhost, this change was needed.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What I am getting at here is: Did we do this because we needed it to run unit tests inside the container, or did we do it because we were trying to use ynr.settings.testing for interactive usage (i.e: things other than running unit tests?)

If the first one, then we need it.
If the second one, then we can revert this.

Comment on lines +5 to +6
DATABASES["default"]["NAME"] = "ynr" # noqa
DATABASES["default"]["USER"] = "ynr" # noqa
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Assuming we're only using testing settings to run tests, you probably don't need this

This didn't work for me. I needed to specify

DATABASES = {
    "default": {
        "ENGINE": "django.db.backends.postgresql",
        "NAME": 'ynr',
        "USER": "ynr",
        "PASSWORD": "ynr",
        "HOST": "dbpsql",
        "PORT": "5432",
    },
}

here. This is with DJANGO_SETTINGS_MODULE=ynr.settings.base in env/frontend.env

Copy link
Member

@chris48s chris48s left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Lower priority things

Comment on lines +32 to +43
FIXME: this doesn't work when run from scratch, with an entirely empty DB.
What schemas need to exist for this to work, and how are they created?

```
Importing parties from The Electoral Commission (using `parties_import_from_ec`)
Traceback (most recent call last):
File "/dc/ynr/venv/lib/python3.8/site-packages/django/db/backends/utils.py", line 89, in _execute
return self.cursor.execute(sql, params)
File "/dc/ynr/venv/lib/python3.8/site-packages/psycopg/cursor.py", line 737, in execute
raise ex.with_traceback(None)
psycopg.errors.UndefinedTable: relation "parties_party" does not exist
LINE 1: ..._party"."ec_data", "parties_party"."nations" FROM "parties_p...
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Based on the error, this looks like maybe you tried to run it without applying migrations first, or possibly because you partially ran the migrations and then they failed due to the issue I flagged with the /data dir. I reckon if you apply that fix and run migrations against a clean DB then running this command should work.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(or at least fail with a different error 😆 )

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll give this a go 👍

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@chris48s @symroe Could you clarify the sequencing that you feel should work for this?

When I start with a freshly created DB instance I then find that I can't run the migrations that @chris48s mentioned above:

$ podman compose down --volumes
...
$ podman compose up -d dbpsql
...
$ ./scripts/container.run.bash python manage.py migrate               
Operations to perform:                                                                                                                                  
  Apply all migrations: (none)                                              
Running migrations:                                                                                                                                     
  No migrations to apply.                                                   
Traceback (most recent call last):                                                                                                                      
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/db/backends/utils.py", line 89, in _execute
    return self.cursor.execute(sql, params)                                                                                                             
  File "/dc/ynr/venv/lib/python3.8/site-packages/psycopg/cursor.py", line 737, in execute
    raise ex.with_traceback(None)                                                                                                                       
psycopg.errors.UndefinedTable: relation "django_content_type" does not exist 
LINE 1: ..."."app_label", "django_content_type"."model" FROM "django_co...

I feel I'm either missing some early schema creation logic somewhere, or I've misunderstood the order of operations. I don't see any fixtures in the YNR repo, as mentioned at https://docs.djangoproject.com/en/4.2/howto/initial-data/, so I assume the migration route that page also mentions is what the project uses - but I don't seem to be able to proceed, as per the above trace.

Any thoughts?

-e CIRCLECI=true \
\
ynr:test \
pytest --ds=ynr.settings.testing -x
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It should not be necessary to pass pytest a flag on the command line to tell it what settings file to use. It should pick this up from the pytest config options in pyproject.toml

DJANGO_SETTINGS_MODULE = "ynr.settings.testing"

container/build/Containerfile Outdated Show resolved Hide resolved
docker-compose.yml Show resolved Hide resolved
docs/INSTALL.md Outdated Show resolved Hide resolved
container/build/system-packages Show resolved Hide resolved
docs/DEVELOPMENT.md Outdated Show resolved Hide resolved
container/build/Containerfile Outdated Show resolved Hide resolved
@chris48s
Copy link
Member

chris48s commented Jan 9, 2025

Sorry. Another one that I didn't catch yesterday but I think this one matters.

Every time I load a page, I get

Internal Server Error: /ajax/ballots/ballots_for_select.json
Traceback (most recent call last):
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/core/handlers/exception.py", line 55, in inner
    response = get_response(request)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/core/handlers/base.py", line 197, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/sentry_sdk/integrations/django/views.py", line 94, in sentry_wrapped_callback
    return callback(request, *args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/views/generic/base.py", line 104, in view
    return self.dispatch(request, *args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/views/generic/base.py", line 143, in dispatch
    return handler(request, *args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/utils/decorators.py", line 46, in _wrapper
    return bound_method(*args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/utils/decorators.py", line 126, in _wrapper_view
    result = middleware.process_request(request)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/middleware/cache.py", line 158, in process_request
    cache_key = get_cache_key(request, self.key_prefix, "GET", cache=self.cache)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/utils/cache.py", line 391, in get_cache_key
    headerlist = cache.get(cache_key)
  File "/dc/ynr/venv/lib/python3.8/site-packages/debug_toolbar/panels/cache.py", line 42, in wrapper
    return panel._record_call(cache, name, original_method, args, kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/debug_toolbar/panels/cache.py", line 149, in _record_call
    value = original_method(*args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/django/core/cache/backends/memcached.py", line 75, in get
    return self._cache.get(key, default)
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/hash.py", line 347, in get
    return self._run_cmd("get", key, default, default=default, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/hash.py", line 322, in _run_cmd
    return self._safely_run_func(client, func, default_val, *args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/hash.py", line 211, in _safely_run_func
    result = func(*args, **kwargs)
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/base.py", line 687, in get
    return self._fetch_cmd(b"get", [key], False, key_prefix=self.key_prefix).get(
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/base.py", line 1133, in _fetch_cmd
    self._connect()
  File "/dc/ynr/venv/lib/python3.8/site-packages/pymemcache/client/base.py", line 424, in _connect
    sock.connect(sockaddr)
ConnectionRefusedError: [Errno 111] Connection refused

in the console. This is using ynr.settings.base for my settings. I think we are basically failing on any operation that tries to use the cache. You will not have hit this using ynr.settings.testing because the test settings just mocks out the cache.

We could get round it for now by just using a mock or local filesystem cache in local dev, but then that's a difference between local and prod. Any bug relating to cache won't reproduce locally. This brings a question about how we're going to approach cache. I guess in our current setup, we just run memcached on the VM (and in local dev) but in a docker-based setup memcached probably needs to become a separate service (ElastiCache?). Ultimately I'd quite like our dev setup to mirror what we're going to do in prod, but also maybe that is something we come to when we look at deployment rather than while we're still doing docker in local dev but VMs in prod.

This might be another one to discuss first.

@chris48s
Copy link
Member

chris48s commented Jan 9, 2025

Having pulled one worm out of the can, as a follow up to that: @symroe - are you aware of anything else that we're running as a service on the VM at the moment aside from memcached?

This adds a Containerfile (a technology-agnostic Dockerfile) that can
successfully install the app's dependencies into a container image. The
image can be built with the following command (note the trailing build
context of "."):

    podman build .

The image's contents are controlled via an allowlist in
.containerignore, which is expected to change significantly in later
commits as more of this repo is used inside the container.

The image is not yet set up to be invoked.
This adds a postgresql compose manifest as a demonstration that
podman-compose can successfully instatiate the service, via:

    podman compose -f container/run/compose.service.postgres.yml up

This is not yet used in any context.
This changes the approach to building the app image by using an Ubuntu
LTS release as the base, and starting by installing the current set of
packages explicitly referenced in deploy/vars.yml (modulo some outdated
entries such as redis-server).

This change is based on discussions with DC personnel with knowledge of
the current setup, and influenced by the likely path we'll take to
adopting a container-based production setup in the future.

The image that's produced is not small, as it contains many, many system
packages that are pulled in as dependencies of the package set we
explicitly specify. The resulting large package set does, however,
permit the app's base Python requirements to be installed successfully,
and this can act as a baseline for future size/efficiency improvements.
This adds a "container" workflow to CircleCI. The workflow is
unoptimised, and initially serves only to test the buildability of the
container image.

The .containerignore file is not respected by CircleCI's Docker
builders, so it is symlinked into place as .dockerignore.
This adds "python manage.py check" to the end of the container image
build process. Whilst this isn't strictly necessary it serves two
useful purposes by:

- demonstrating that the container *can* run the check; and
- pre-generating many .pyc files and including them in the image for
  faster app initialisation.

The Containerfile is also tweaked to:

- read the list of system packages from an external file;
- install the app's python requirements before copying its code, so the
  DX happy path doesn't incur the pip-install cost when only modifying
  code;
- install the app's python requirements based on the sopn_parsing.txt
  file as this transitively includes "pandas", without which the check
  command fails.
This adds a simplistic mechanism to invalidate CircleCI's Docker Layer
Cache (DLC), and force an image rebuild from scratch.

Locally, "podman build --no-cache ." should be preferred instead.
Invoke a cut-down version of the current CI test command inside the
built container.
This adds some system and pip dependencies that are currently encoded in
the CI config. The system dependencies are copied into the container's
package file, but a manually fetched system dependency (pandoc) is
translated into its convenience pip package equivalent.

The change to pandoc's installation mechanism requires the use of an
alternative version of the pypandoc library that also provides the
underlying pandoc binary. This requires us to use a later version than
the currently-used pypandoc version, as the currently-used version
doesn't have a binary-including-package counterpart. We bump to the
latest pip package version.
This adds a docker-compose.yml file which, along with the included changes,
permits a developer to run a hot-reloading instance of the app.

To make this work, it:
- invokes `npm install` early in the image build process
- adds gunicorn to requirements/base.txt (as there aren't any environments
  where this is optional)
- makes some (hopefully harmless!) changes to ynr/settings/testing.py that
  allow the app to start successfully

This change also:
- adds a convenience script at scripts/container.image.build.bash, intended for
  use both locally and in CI
- increases the commenting/structure in the Containerfile

The compose file must be at the repo root (and not inside the defunct
container/run/ directory) because relative bind mount sources are resolved
relative to the compose file's location, and we need to mount the ynr/
directory from inside the repo root. This Docker (hence also Podman) feature
even goes so far (or appears to!) as to resolve the location reading through a
symlink, so we can't simply link a compose file sitting at the repo root to a
file in a subdirectory.

The developer currently needs to manually execute some asset-related commands
before starting the compose stack, in order to make app render correctly. These
commands place content into the ynr/ directory, which is bind-mounted from the
developer's machine. They will be automated in a later change:

    podman compose run --rm --no-deps -e DJANGO_SETTINGS_MODULE=ynr.settings.testing frontend npm run build
    podman compose run --rm --no-deps -e DJANGO_SETTINGS_MODULE=ynr.settings.testing frontend python manage.py collectstatic --no-input
This cleans the NPM cache after installation, and also uses the more
deterministic "npm ci" command instead of "npm install".

Also:
- help future beginners by defining SOPN in the top-level README
- reduce top-level repo clutter by renaming .containerignore to .dockerignore.
This adds a convenience shim for invoking "python manage.py" inside a freshly
instantiated container, connected to the same DB and bind mounts that the
composed "frontend" container would be able to access:

    $ ./scripts/container.compose.manage-py.bash check 2>/dev/null
    WARNING: no local settings file found. See local.py.example
    System check identified no issues (0 silenced).

Invocations are currently noisy (hence the 2> redirection); this will be
improved later.
This expands the set of file and directory paths which are bind mounted into
the frontend container when using compose. The set is aligned with the paths
included in the built container image via the .dockerignore file's contents.
This permits the container image required by the local compose stack to be
built directly using the compose subcommand:

    podman compose build
or
    podman compose build frontend

A test is added to CI that asserts compose is superficially happy. We don't
(yet) use compose to run any containers in CI, so this step is necessary
trivial.

Also: fix a podman/docker inconsistency where podman permitted compose "ports"
to be a scalar value, but docker requires a list of values.
This allows a developer to place a file at the gitignored path env/frontend.env
containing VAR=val key/value pairs which the frontend compose container will
read on startup and instantiate as environment variables.

This mechanism has been chosen as it affects all frontend container
invocations, including those that don't start Django as a web server. This
consistency will reduce unexpected drift between the primary frontend container
started by a developer and any secondary containers.

There are complications that currently make it fiddly to get this file *also*
bind-mounted as the .env file that ynr/wsgi.py loads. For the moment, this
means that environment variable changes require a restart of the frontend
container.
This adds a script that shims "podman compose exec", and changes the renamed
shim script that invokes Django management commands to use it.

It also removes a couple of envvars from the compose file which are more
properly situated in the gitignored, per-developer env/frontend.env file.
/ynr/media/ doesn't need to be included inside the frontend container image.
YNR's production RDS runs v16.4. Both the local containerised setup and the
in-CI DB should mirror this.
This adds a script that executes pytest inside a running frontend container,
passing any parameters through to pytest.
Docker compose requires an explicit healthcheck, whereas podman appears to
infer a healthy container when an exposed port is listening.
This reverts commit 4af80b5.

This was added as docker required it, so that we could test using
docker-compose in CI. This has proven impractical, given the warnings at
https://circleci.com/docs/docker-compose/#using-docker-compose-with-docker-executor:

    The remote docker daemon runs on a different system than the docker CLI and
    docker compose, so you must move data around to make this work. Mounting
    can usually be solved by making content available in a docker volume.

In order to make this work, we'd have to vary the compose file so that in local
dev content was bind-mounted, whereas it would be made available in a volume in
CI. This would negate the point of running the compose stack in CI: to prove
that local-dev workflows are happy.

Because podman appears to have a problem with this setup on certain machines:

    ERRO[0000] Failed to start transient timer unit: Unit
    cdb85bbc93bfd59cc38703842f96c12d0cced997a7ee582b2954a6bb87905804.timer was
    already loaded or has a fragment file.

... our easiest route forward is to revert this change until we genuinely need
to solve docker's healthcheck problem.
This adds scripts/container.run.bash, which runs a command in a fresh frontend
container.

Also: fix trailing whitespace
This allows a developer who invokes scripts/container.image.build.bash to
specify arbitrary parameters for the underlying builder.

e.g. "--no-cache", which is a better way of avoiding using the local build
cache than bumping the CI-focussed build ARG at the top of
container/build/Containerfile.
This adds some docs to help a developer get started modifying and running YNR
inside a container.

Also: structure README.md's headings more hierarchically, leaving only a single
H1 on the page; some light rewording in README.md to improve its flow, along
with a short new section that points a member of the public towards
whocanivotefor.co.uk if they happen to come across this repo.
@jpluscplusm jpluscplusm force-pushed the jcm/add-local-dev-container-workflows branch from a169175 to b2b4d5c Compare January 9, 2025 10:51
Initial development of containerised YNR used podman-compose v1.0.6, but that
version is incompatible with the current schema used by the docker-compose.yml
file. v1.2.0 is noted as being specifically required as the very recent v1.3.0
has not yet been properly tested with this branch.
@symroe
Copy link
Member

symroe commented Jan 9, 2025

On cache: I use the dummy cache locally, but I suspect the easiest thing to do here is to switch to DatabaseCache. We don't really need a dedicated caching service for this app.

Some DB migrations require files that exist in the data directory. This change
ensures that it exists in the container image by adding it to the .dockerignore
include list, and also bind mounts the path so that developers can update its
files without rebuilding the image.
@chris48s
Copy link
Member

chris48s commented Jan 9, 2025

Just a note on ways of working:

With a big PR like this where there will be multiple rounds of comments and changes, I prefer it if we don't rebase and force-push the branch mid-review.

The reason for this is: If we push additional commits to address review comments, it is easy for the reviewer to come back to a PR and quickly see what has changed since they last looked at it. The reviewer can easily tell from GitHub's UI that they've reviewed everything up to commit a169175 (for example) so everything after that is new since the last round of comments. If the entire history has been re-written since they last viewed the PR and new changes are squashed into commits further back in history, it is much harder to tell at a glance "what changed since I last looked at this?". The reviewer basically has to come back to the thing from first principles.

It doesn't matter that we've done a rebase and force-push already. Lets just play the ball where it landed from this point, but for subsequent changes, can we just push more commits to the branch and then squash things to tidy up as a final step once we're ready to merge.

Thanks

@jpluscplusm
Copy link
Collaborator Author

With a big PR like this where there will be multiple rounds of comments and changes, I prefer it if we don't rebase and force-push the branch mid-review.

Understood. The rebase was solely to remove the first-and-last commits' mitigations for a failing test that magically fixed itself on tip of master. I'm already adding commits as requested and won't force push any more rebases.

Some editors complain if a Containerfile doesn't use uppercase instructions.
This tweaks some docs following PR feedback.
@jpluscplusm
Copy link
Collaborator Author

On cache: I use the dummy cache locally, but I suspect the easiest thing to do here is to switch to DatabaseCache. We don't really need a dedicated caching service for this app.

Whilst I can configure the webapp to use DatabaseCache, I feel that changing this setting (specifically where it's configured, and how we nudge the developer to alter it if required) should come after we've had the broader settings-related discussion that @chris48s suggested above. Thoughts?

This alters the Containerfile's setting of NODE_ENV so it's less susceptible to
future line-order changes having unintended side effects.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants