How to manage permissions for a volume mounted into a docker container?



  • I'm developing a Wordpress theme, and want to use Docker in my dev setup. What I've done is pretty simple:

    • create a database service running MySQL 5.7
    • create a wordpress service, where I mount my theme folder as a volume into /var/www/html/wp-content/themes

    I'm struggling with the volume permissions, however.

    I'm working with the following project folder structure:

    .
    ├── docker-compose.yml
    └── my-theme
    
    

    My docker-compose.yml file looks like this:

    version: '3.2'
    
    services:
      database:
        image: mysql:5.7
        volumes:
          - my_data:/var/lib/mysql
        restart: always
        environment:
          MYSQL_ROOT_PASSWORD: root
          MYSQL_DATABASE: wordpress
          MYSQL_USER: wordpress
          MYSQL_PASSWORD: root
    
      wordpress:
        depends_on:
          - database
        image: wordpress:php7.3-apache
        ports:
          - '8000:80'
        restart: always
        environment:
          WORDPRESS_DB_HOST: database:3306
          WORDPRESS_DB_USER: wordpress
          WORDPRESS_DB_PASSWORD: root
        working_dir: /var/www/html
        volumes:
          - type: volume
            source: ./my-theme
            target: /var/www/html/wp-content/themes/my-theme
    volumes:
      my_data: {}
    
    

    When I run docker-compose up, everything works as expected: containers are created, and I can access Wordpress in the browser. However, the theme I mounted as a volume doesn't render anything when I activate it.

    When I sh into the wordpress container (docker-compose exec wordpress sh) I can see that the wp-content/themes folder is owned by root. So I figured that was the problem.

    I verified this being a permissions issue by manually and recursively chowning the wp-content folder in the container:

    <pre class="lang-sh prettyprint-override">```
    chown -R www-data:www-data /var/www/html/wp-content
    
    
    
    Once that was done, my theme rendered as expected. So now I'm looking for a way to avoid this `chown` process (the idea is that any other dev can clone this project, simply run `docker-compose up` and start working).
    
    The first thing I tried was to make a **Dockerfile** where I would build a slightly customized Wordpress image:
    
    

    FROM wordpress:php7.3-apache
    RUN mkdir -p /var/www/html/wp-content/themes/test-theme
    && chown -R /var/www/html/wp-content

    
    My reasoning behind this was that by creating the directory and `chown`ing it beforehand, the volume would inherit the user:group mapping. Alas, no such thing; mounting the volume overrides this mapping and sets it back to `root:root`.
    
    After that, I tried to set the `APACHE_RUN_USER` and `APACHE_RUN_GROUP` environment variables in my `docker-compose.yml` file:
    
    

    version: '3.2'

    services:
    database:
    ...

    wordpress:
    ...
    environment:
    WORDPRESS_DB_HOST: database:3306
    WORDPRESS_DB_USER: wordpress
    WORDPRESS_DB_PASSWORD: root
    APACHE_RUN_USER: '$(id -u)'
    APACHE_RUN_GROUP: '$(id -g)'
    working_dir: /var/www/html
    volumes:
    - type: volume
    source: ./my-theme
    target: /var/www/html/wp-content/themes/my-theme
    volumes:
    my_data: {}

    
    However, this threw a bunch of apache errors on build.
    
    I'm at a bit of a loss here now. Is there any best practice for managing permissions of mounted volumes in Docker? I've googled a lot for this, but the solutions I do find go a little bit over my head.


  • You can do this by overwriting the entrypoint for the wordpress image.

    Create a file startup.sh in your project and make is executable:

    #!/bin/bash
    
    chown -R www-data:www-data /var/www/html/wp-content
    docker-entrypoint.sh apache2-foreground
    
    

    Then in your docker-compose.yml:

    ...
      wordpress:
    ...
        working_dir: /var/www/html
        volumes:
            - './my-theme:/var/www/html/wp-content/themes/my-theme'
            - './startup.sh:/startup.sh'
        entrypoint: /startup.sh
    
    

    This worked for me, let me know if you have problems implementing it.



最新帖子

最新内容

  • S

    Listen for the dragStart event :

    div.find("canvas").on('dragstart', function (event) {

    so you can use DataTransfer.setDragImage() :

    When a drag occurs, a translucent image is generated from the drag target (the element the dragstart event is fired at), and follows the mouse pointer during the drag. This image is created automatically, so you do not need to create it yourself. However, if a custom image is desired, the DataTransfer.setDragImage() method can be used to set the custom image to be used. The image will typically be an element but it can also be a or any other image element.

    And update the container.onDragStart function to :

    container.onDragStart = function (evt) { container.selected(evt); prevX = evt.clientX; prevY = evt.clientY; evt.originalEvent.dataTransfer.setDragImage(null, 0, 0); // add this line };

    Here's an example with a custom " shadow" iamge :

    <pre class="snippet-code-js lang-js prettyprint-override">``` $(document).ready(function() { // dont have a webserver so im using base64string instead var maskedImageUrlb = ""; // maskedImage two var mask2 = $(".container").mask({ maskImageUrl: maskedImageUrlb, onMaskImageCreate: function(img) { // add your style to the img example below img.css({ "position": "fixed", "left": 173, "top": 1 }) } }); fileupa2.onchange = function() { mask2.loadImage(URL.createObjectURL(fileupa2.files[0])); }; }); // end of document ready // jq plugin for mask (function($) { var JQmasks = []; $.fn.mask = function(options) { // This is the easiest way to have default options. var settings = $.extend({ // These are the defaults. maskImageUrl: undefined, imageUrl: undefined, scale: 1, id: new Date().getUTCMilliseconds().toString(), x: 0, // image start position y: 0, // image start position onMaskImageCreate: function(div) {}, }, options); var container = $(this); let prevX = 0, prevY = 0, draggable = false, img, canvas, context, image, timeout, initImage = false, startX = settings.x, startY = settings.y, div; container.mousePosition = function(event) { return { x: event.pageX || event.offsetX, y: event.pageY || event.offsetY }; } container.selected = function(ev) { var pos = container.mousePosition(ev); var item = $(".masked-img canvas").filter(function() { var offset = $(this).offset() var x = pos.x - offset.left; var y = pos.y - offset.top; var d = this.getContext('2d').getImageData(x, y, 1, 1).data; return d[0] > 0 }); JQmasks.forEach(function(el) { var id = item.length > 0 ? $(item).attr("id") : ""; if (el.id == id) el.item.enable(); else el.item.disable(); }); }; container.enable = function() { draggable = true; $(canvas).attr("active", "true"); div.css({ "z-index": 2 }); } container.disable = function() { draggable = false; $(canvas).attr("active", "false"); div.css({ "z-index": 1 }); } container.onDragStart = function(evt) { container.selected(evt); prevX = evt.clientX; prevY = evt.clientY; var img = new Image(); img.src = 'https://www.what-dog.net/Images/faces2/scroll001.jpg'; evt.originalEvent.dataTransfer.setDragImage(img, 10, 10); }; container.getImagePosition = function() { return { x: settings.x, y: settings.y, scale: settings.scale }; }; container.onDragOver = function(evt) { if (draggable && $(canvas).attr("active") === "true") { var x = settings.x + evt.clientX - prevX; var y = settings.y + evt.clientY - prevY; if (x == settings.x && y == settings.y) return; // position has not changed settings.x += evt.clientX - prevX; settings.y += evt.clientY - prevY; prevX = evt.clientX; prevY = evt.clientY; container.updateStyle(); } }; container.updateStyle = function() { clearTimeout(timeout); timeout = setTimeout(function() { context.clearRect(0, 0, canvas.width, canvas.height); context.beginPath(); context.globalCompositeOperation = "source-over"; image = new Image(); image.setAttribute('crossOrigin', 'anonymous'); image.src = settings.maskImageUrl; image.onload = function() { canvas.width = image.width; canvas.height = image.height; context.drawImage(image, 0, 0, image.width, image.height); div.css({ "width": image.width, "height": image.height }); }; img = new Image(); img.src = settings.imageUrl; img.setAttribute('crossOrigin', 'anonymous'); img.onload = function() { settings.x = settings.x == 0 && initImage ? (canvas.width - (img.width * settings.scale)) / 2 : settings.x; settings.y = settings.y == 0 && initImage ? (canvas.height - (img.height * settings.scale)) / 2 : settings.y; context.globalCompositeOperation = 'source-atop'; context.drawImage(img, settings.x, settings.y, img.width * settings.scale, img.height * settings.scale); initImage = false; }; }, 0); }; // change the draggable image container.loadImage = function(imageUrl) { if (img) img.remove(); // reset the code. settings.y = startY; settings.x = startX; prevX = prevY = 0; settings.imageUrl = imageUrl; initImage = true; container.updateStyle(); }; // change the masked Image container.loadMaskImage = function(imageUrl, from) { if (div) div.remove(); canvas = document.createElement("canvas"); context = canvas.getContext('2d'); canvas.setAttribute("draggable", "true"); canvas.setAttribute("id", settings.id); settings.maskImageUrl = imageUrl; div = $("<div/>", { "class": "masked-img" }).append(canvas); div.find("canvas").on('dragstart', function(event) { if (event.handled === false) return; event.handled = true; container.onDragStart(event); }); div.find("canvas").on('touchend mouseup', function(event) { if (event.handled === false) return; event.handled = true; container.selected(event); }); div.find("canvas").bind("dragover", container.onDragOver); container.append(div); if (settings.onMaskImageCreate) settings.onMaskImageCreate(div); container.loadImage(settings.imageUrl); }; container.loadMaskImage(settings.maskImageUrl); JQmasks.push({ item: container, id: settings.id }) return container; }; }(jQuery));

    <pre class="snippet-code-css lang-css prettyprint-override">```
    .container {
    border: 1px solid #DDDDDD;
    display: flex;
    background: red;
    width: 612px;
    height: 612px;
    }

    .container canvas {
    display: block;
    }

    .masked-img {
    overflow: hidden;
    margin-top: 50px;
    position: relative;
    }

    <pre class="snippet-code-html lang-html prettyprint-override">``` <script src="https://code.jquery.com/jquery-3.3.1.min.js"> </script> image 2 <input id="fileupa2" type="file"> <div class="container"> </div>

    read more
  • S

    When user upload image and when he try to drag the image , the image will display shadow along with dragging, but i don't want to display the shadow....

    codepen : https://codepen.io/kidsdial/pen/xMNbVz

    <pre class="snippet-code-js lang-js prettyprint-override">``` $(document).ready(function () { // dont have a webserver so im using base64string instead var maskedImageUrlb =""; // maskedImage two var mask2 = $(".container").mask({ maskImageUrl: maskedImageUrlb, onMaskImageCreate: function (img) { // add your style to the img example below img.css({ "position" : "fixed" , "left": 173, "top": 1 }) } }); fileupa2.onchange = function () { mask2.loadImage(URL.createObjectURL(fileupa2.files[0])); }; }); // end of document ready // jq plugin for mask (function ($) { var JQmasks = []; $.fn.mask = function (options) { // This is the easiest way to have default options. var settings = $.extend({ // These are the defaults. maskImageUrl: undefined, imageUrl: undefined, scale: 1, id: new Date().getUTCMilliseconds().toString(), x: 0, // image start position y: 0, // image start position onMaskImageCreate: function (div) { }, }, options); var container = $(this); let prevX = 0, prevY = 0, draggable = false, img, canvas, context, image, timeout, initImage = false, startX = settings.x, startY = settings.y, div; container.mousePosition = function(event){ return { x: event.pageX || event.offsetX, y: event.pageY || event.offsetY }; } container.selected = function (ev) { var pos = container.mousePosition(ev); var item =$(".masked-img canvas").filter(function(){ var offset = $(this).offset() var x = pos.x - offset.left; var y = pos.y - offset.top; var d = this.getContext('2d').getImageData(x, y, 1, 1).data; return d[0] >0 }); JQmasks.forEach(function(el){ var id = item.length> 0 ? $(item).attr("id") : ""; if (el.id ==id ) el.item.enable(); else el.item.disable(); }); }; container.enable = function(){ draggable = true; $(canvas).attr("active", "true"); div.css({ "z-index": 2 }); } container.disable = function(){ draggable = false; $(canvas).attr("active", "false"); div.css({ "z-index": 1 }); } container.onDragStart = function (evt) { container.selected(evt); prevX = evt.clientX; prevY = evt.clientY; }; container.getImagePosition = function () { return { x: settings.x, y: settings.y, scale: settings.scale }; }; container.onDragOver = function (evt) { if (draggable && $(canvas).attr("active") === "true") { var x = settings.x + evt.clientX - prevX; var y = settings.y + evt.clientY - prevY; if (x == settings.x && y == settings.y) return; // position has not changed settings.x += evt.clientX - prevX; settings.y += evt.clientY - prevY; prevX = evt.clientX; prevY = evt.clientY; container.updateStyle(); } }; container.updateStyle = function () { clearTimeout(timeout); timeout = setTimeout(function () { context.clearRect(0, 0, canvas.width, canvas.height); context.beginPath(); context.globalCompositeOperation = "source-over"; image = new Image(); image.setAttribute('crossOrigin', 'anonymous'); image.src = settings.maskImageUrl; image.onload = function () { canvas.width = image.width; canvas.height = image.height; context.drawImage(image, 0, 0, image.width, image.height); div.css({ "width": image.width, "height": image.height }); }; img = new Image(); img.src = settings.imageUrl; img.setAttribute('crossOrigin', 'anonymous'); img.onload = function () { settings.x =settings.x == 0 && initImage ? (canvas.width - (img.width * settings.scale )) / 2 : settings.x; settings.y =settings.y == 0 && initImage ? (canvas.height - (img.height * settings.scale )) / 2 : settings.y; context.globalCompositeOperation = 'source-atop'; context.drawImage(img, settings.x, settings.y, img.width * settings.scale, img.height * settings.scale); initImage = false; }; }, 0); }; // change the draggable image container.loadImage = function (imageUrl) { if (img) img.remove(); // reset the code. settings.y = startY; settings.x = startX; prevX = prevY = 0; settings.imageUrl = imageUrl; initImage = true; container.updateStyle(); }; // change the masked Image container.loadMaskImage = function (imageUrl, from) { if (div) div.remove(); canvas = document.createElement("canvas"); context = canvas.getContext('2d'); canvas.setAttribute("draggable", "true"); canvas.setAttribute("id", settings.id); settings.maskImageUrl = imageUrl; div = $("<div/>", { "class": "masked-img" }).append(canvas); div.find("canvas").on('touchstart mousedown', function (event) { if (event.handled === false) return; event.handled = true; container.onDragStart(event); }); div.find("canvas").on('touchend mouseup', function (event) { if (event.handled === false) return; event.handled = true; container.selected(event); }); div.find("canvas").bind("dragover", container.onDragOver); container.append(div); if (settings.onMaskImageCreate) settings.onMaskImageCreate(div); container.loadImage(settings.imageUrl); }; container.loadMaskImage(settings.maskImageUrl); JQmasks.push({item: container , id: settings.id}) return container; }; }(jQuery));

    <pre class="snippet-code-css lang-css prettyprint-override">```
    .container {
    border: 1px solid #DDDDDD;
    display: flex;
    background: red;
    width:612px;
    height:612px;
    }

    .container canvas {
    display: block;
    }

    .masked-img {
    overflow: hidden;
    margin-top: 50px;
    position: relative;
    }

    <pre class="snippet-code-html lang-html prettyprint-override">``` <script src="https://code.jquery.com/jquery-3.3.1.min.js"> </script> image 2 <input id="fileupa2" type="file" > <div class="container"> </div> I don't want to display shadow when dragging.... Please let me know if you have any doubts on this.... Thanks in Advance....

    read more
  • S

    I am getting this error when getting address using Geocoder in react native:

    Object { "code": 4, "message": "Error from the server while geocoding. The received datas are in the error's 'origin' field. Check it for more informations.", "origin": Object { "results": Array [], "status": "ZERO_RESULTS", }, }

    I am using "react-native-geocoding" dependency

    My code include in here:

    Geocoder.init(GOOGLE_API_KEY); //google api key Geocoder.from(latitude,longitude) .then(json => { var addressComponent = json.results[0].address_components; console.log(JSON.stringify(addressComponent)); }) .catch(error => console.warn(error));

    Need some help?

    read more
  • S

    I want to use load vdi file into windows hyper v.I can convert vdi to vdh and use in hyper v.But I want to use vdi file in hyper v. Any information, please?

    read more
  • S

    No, there is no problem in communicating directly to the servers.

    Consul clients are used in big data centers with many (5+) Consul agents. The Consul developers recommend to use three to five server agents per datacenter. If you need more agents (for hundreds of micro services e.g.) than you should use client agents that are connected to server agents instead of launching more server agents which will decrease performance.

    But in a smaller datacenter there is no problem using server agents directly.

    read more

推荐阅读