For reference, I found this closed issue when creating this one: #120 (closed)
We have a couple of developers here developing on Apple Silicon. The step by step instructions work out of the box except for the a d4d up step, where the following type of issue recurs for multiple required service images:
Error response from daemon: image with reference [service/name:x] was found but does not match the specified platform: wanted linux/amd64, actual: linux/arm64
To work around this I created a docker-compose.override.yml file in the project root, where I copied the entire contents of the main docker-compose.yml file (which gets overwritten on composer install) and added the platform: linux/amd64 line before the image for the services in question, namely mariadb, mariadbtest, pma and mailhog, eg:
So the linux/amd64 version of the images in question are explicitly requested. At some point I expect arm64 variants will be released and they will have better performance, so I wouldn't suggest committing this fix in the repo. But this could be useful on a FAQ/Troubleshooting/Documentation page for Apple Silicon users.
Which did the trick, but there was one remaining issue, which is the 'Mounts denied' issue, caused by the '/drupal:/ssh-agent' line in the php.volumes list. To fix this (without editing docker-compose.yml directly), I recreated the docker-compose.override.yml file with the following:
Going back to your reply, it would be great to not have to maintain multiple docker-compose files, but entering the above in the .lakedrops.user.yml file just merges the keys (possibly related to this issue?). If there's another workaround, I'd be happy to try it, otherwise, this works for now
there was one remaining issue, which is the 'Mounts denied' issue, caused by the '/drupal:/ssh-agent' line in the php.volumes list.
@rominronin I'd be very interested to fix this properly. There had been a similar issue before, which we had to close due to missing feedback, see #127 (closed). The SSH_AUTH_SOCK environment variable should be forwarded from the host into the L3D container and may have to be altered to work properly in all context. If you had some time to debug this, where this breaks on MacOS, that would be great and I'd be really happy to fix this in the code.
@kaigertz this may be of interest to you as well as you had reported a similar issue in #120 (closed) before.
"The SSH_AUTH_SOCK environment variable should be forwarded from the host into the L3D container and may have to be altered to work properly in all context. If you had some time to debug this, where this breaks on MacOS, that would be great and I'd be really happy to fix this in the code."
I'm not sure this is the issue. The line in the default docker-compose.yml file is which causes the error is:
php: volumes: - '/drupal:/ssh-agent'
It looks like the path '/drupal' is expected in the local machine (because the mapping definitions are from:to). I assume this works on linux/windows environments (but I can check later on a colleague's PC).
It's when this line is replaced with '${SSH_AUTH_SOCK}:/ssh-agent' that it works on macos.
So what is this line for? I'm reading tonnes of documentation to find out. It's not in the default docker4drupal docker-compose.yml file, that's for sure.
It may actually pretty straight forward by just following the execution of the code to find out where this gets the value /drupal from by mistake and why.
If ($this->isCiContext() || $this->isLocalDevMode()) returns false, we get the desired value of $SSH_AUTH_SOCK. Conversely, the current value of /drupal is assigned if the context is CI/CD OR a local dev environment (so in all cases? What other contexts are there? Is this a mistake?).
I'm trying to understand what this statement is for. It looks like the default behaviour should be to assign
$php['ssh_auth_sock'] = '$SSH_AUTH_SOCK';
and ONLY assign something else IF an alternative mount point is provided by docker.
So perhaps the code should instead read as follows:
if ($php['ssh']) { $php['ssh_auth_sock'] = $this->getDockerMountSource('/ssh-agent') ?? '$SSH_AUTH_SOCK'; }
with the additional change to getDockerMountSource() (just returns NULL if all else fails - this is where /drupal is ACTUALLY coming from):
private function getDockerMountSource($projectRoot): string|null { $currentDir = getcwd(); $container = $this->readContainerConfig(); foreach ($container['Mounts'] as $mount) { if (empty($projectRoot)) { if ($currentDir === $mount['Destination']) { return $mount['Source']; } } else if (strpos($projectRoot, $mount['Destination']) === 0) { return $mount['Source'] . substr($projectRoot, strlen($mount['Destination'])); } } return NULL; }
But this is based on guessing the purpose of this statement.
Note: I initially went another route: $this->isCiContext() returns false, and $this->isLocalDevMode() just returns the LAKEDROPS_DEV_ENV env variable. So I can just edit that, right?
is assigned if the context is CI/CD OR a local dev environment (so in all cases? What other contexts are there? Is this a mistake?).
There is also the LIVE or production mode. I don't remember, why it was necessary to make that distinction, though.
change to getDockerMountSource() (just returns NULL if all else fails - this is where /drupal is ACTUALLY coming from)
It's not that simple. Let me show you the situation on a Linux host:
On the host, SSH_AUTH_SOCK is /run/user/1000/keyring/ssh
In the L3D container, SSH_AUTH_SOCK is /ssh-agent which is mounted to /run/user/1000/keyring/ssh on the host
In the PHP container, we need to mount to /run/user/1000/keyring/ssh as well, so the docker compose config needs to have this volume: /run/user/1000/keyring/ssh:/ssh-agent
Would we use the value of the $SSH_AUTH_SOCK variable inside the L3D container when building and starting the docker compose project, then would mount to /ssh-agent in L3D, which is wrong. Since Docker operates on the host, the PHP container also has to mount to the correct directory on the host level.
That's what getDockerMountSource('/ssh-agent') in there for. It analyses the current L3D container and finds the mount point for that /ssh-agent directory.
It looks like, it doesn't find that, which seems to be a MacOS issue. If you could help to find the reason why getDockerMountSource is not finding the correct mount point in your environment, that would actually fix the problem.
I think we have a solution.
Can you move to the buildx infastructure in order to build amd and arm images.
example docker buildx build --load --build-arg VERSION=${VERSION} --platform linux/amd64,linux/arm64 -t ${CI_REGISTRY_IMAGE}/run:${VERSION} ./run
this would solve multiple problems for apple silicone macs we hope.
so no need for a overwrite in the composer file etc and it is just an additional images and all the old images are still working. buildx will replace build anyway in the future anyway.
@nico-acolono this sounds promising. And we already use the buildx framework as the build process happens inside an image of the docker/gitlab-drupal-ci project which is configured with docker buildx install to declare docker build as an alias for docker buildx build.
However, inside of that context we can't use the multi-platform build. That shouldn't be a problem as we could easily build twice, once for each platform. Just wondering about the image tagging, any idea how that should look like then?
It didn't work. Unfortunately, I didn't note the error message back then. But building the image wasn't the only issue, it's also the question about tagging then. I just don't know how to do all that and need input.
Well, we know the syntax on how it is supposed to be working. You can see that in the comments above. But it's not building. The error message is actually in the next comment below this one.
No matter how much I reduce the Dockerfile, when trying to build for the linux/arm64 platform, we get the exec /bin/sh: exec format error exception. Is that really the correct platform string?
To read on why you are getting the dreaded exec /bin/sh/: exec format error, please visit this post. To work around this issue, I installed UTN virtualisation software on my MacBook Pro M2 and installed Ubuntu Desktop 64-bit in it as an emulator. Ubuntu needs to be installed as an emulator because we are trying to emulate x64 on an arm system.
thx for your input, but I would still aim for a solution, where I don't need an extra visualization tool.
main reason, to have everything in docker is (in my opinion) that it works on every system. for sure not easy, but I would aim for that :P
Thank you!
It must be possible, to get Docker and our images to run on Mac natively. It's just that we need to find out, what needs to be done. I hope, somebody will eventually find out.
@jurgenhaas in that post, it states that docker image must be rebuilt for arm, but I honestly have no idea if we can use image by platform, i.e. if it's arm get this image or if it's x64 then get this image.
There have been some comments above on how to build images that are supporting multiple platforms, or to have separate images per platform. I had some tests around that and open questions, but then the issue got stuck. Happy to follow-up on that, if we find the necessary input. @rominronin maybe you can get back in on that one?
Well, it looks like it builds OK on a Mac infrastructure, but that doesn't help because we can't run CI pipelines on a Mac. We need to build those images on Alpine or Ubuntu hardware. The question, how can that be done?
I have another question however: what does this build process achieve? Because after the above build, I updated docker desktop, recreated my containers for a lakedrops enabled project: and everything works again!
What was built with the above commands and do I have that process to thank for restoring my local (l3d) development environment?
Note this is with the caveat that I still need the docker-compose.override and .lakedrops.user files mentioned at the top of this thread
What we're working on here is to get the L3D images to a state where they can also work on Apple Silicon. Building the image as in the step 2 days ago, doesn't do anything to any of the working solutions. It's only a test on what we need to do to get the L3D images build for linux/arm64. Once that's resolved, we want to tale that into the L3D projects so that future versions of this project can use them.
In other words, I'm asking for help to get the images built in the pipeline. And it was a suggestion from acolono to build the L3D images with tha flag --platform linux/amd64,linux/arm64, but that doesn't work. This is what we need to get fixed first. Once that's done, I'm ready to go the second step. But there is no point in discussing a dozen different steps, if we're not able to resolve the first one.
So, who can help us to get our images build for linux/amd64 and linux/arm64 at the same time? Is that even necessary?
As for your requirement of a fully copied docker-compose.override file, we already reached the point that you don't need that when you add the correct lines to your .lakedrops.yml to extend the automatically created docker-compose.yml file
The follow-up problem with the wrong value /drupal:/ssh-agent' for the SSH agent environment variable, there is no point in doing workarounds. We need to find out, why the docker4drupal composer plugin or the L3D container makes that mistake. But since that error only happens on Mac, I can't find this out myself. Somebody needs to debug this to death. WE NEED TO FIND OUT THE SOURCE OF THE PROBLEM - we don't need workarounds.
My current mood is this: is there anybody to do all the research and debugging on the Mac now? If so, I'm happy to help. If not, I will once and forever declare that L3D does not support the Mac platform. This is now going on for 2 years, and I have no benefit from this whatsoever. I give it another week, otherwise I'm done with this.
Thank you Jürgen.
As discussed in our chat, I will gather and collect current infos and go through it. I will have a Meeting With Baris and Dave later this month and try to figure out steps, to tackt the issue and provide possible solutions in the end. Please ignore updates in this issue until one is mentioning you directly, so you can focus on onther tasks and you don't get distracted from this one.
Looking forward to get this issue closed somehow.
Nico