For over 5+ years we help companies reach their financial and branding goals. oDesk Software Co., Ltd is a values-driven technology agency dedicated

Gallery

Contacts

Address

108 Tran Dinh Xu, Nguyen Cu Trinh Ward, District 1, Ho Chi Minh City, Vietnam

E-Mail Address

info@odesk.me

Phone

(+84) 28 3636 7951

Hotline

(+84) 76 899 4959

Websites Management

NGINX Unit 1.18.0 Adds Filesystem Isolation and Other Enhancements

Usually, we’d start a blog post like this one rather light‑heartedly, but the mood of the times is undoubtedly somber, the reasons for that being too numerous to joke about. However, we hope you’re doing well – and again, we have some news to share.

For starters, we at the NGINX Unit team believe that the term “isolation” doesn’t necessarily deserve the bad rap it has gained recently around the globe, and our most recent release, NGINX Unit 1.18.0, has arrived with an accordingly themed major update.

Filesystem Isolation

The isolation family of application settings, introduced in NGINX Unit 1.11.0, now includes a new rootfs object. If the underlying OS allows it, you can use rootfs to designate an arbitrary directory as the filesystem root.

{
“type”: “python 2.7”,
“path”: “/”,
“home”: “/venv/”,
“module”: “wsgi”,
“isolation”: {
“rootfs”: “/var/app/sandbox/”
}
}

view rawrootfs hosted with ❤ by GitHub

An application configured this way is locked within the confines of the /var/app/sandbox/ directory, unable to access any files or directories outside of it. All path settings that you provide for the app need to account for that, as illustrated by the path and home objects in the example above.

The potential applications for this capability, especially combined with other isolation features, are significant. Effectively, it enables you to configure and run apps as lightweight on‑demand containers, improving their security, isolating them from each other and the underlying OS, and enhancing the granularity of your infrastructure. That is exactly the goal of our containerization initiative, and isolation of the filesystem root is an important milestone in our endeavor.

While isolating your app, NGINX Unit conveniently ensures the appropriate language runtime is still available for all languages we support. Moreover, you can use rootfs to maintain different versions of your app’s runtime (modules, libraries, and so on) and toggle between them seamlessly. An appropriately detailed hands‑on demonstration is beyond the scope of this blog post, so keep an eye out for an upcoming blog dedicated to the topic. Meanwhile, you can try this addition to see the benefits it can bring to your own infrastructure.

For further details, see our Process Isolation configuration guide.

PHP Targets

The second innovation introduced in NGINX Unit 1.18.0 is specific to PHP. It aims to simplify the daily work of the PHP-on-Unit crowd who previously had to maintain rather cumbersome configurations for their beloved apps.

The NGINX Unit team initially noticed that more often than not PHP web apps require several modes of operation for the different ways that individual scripts within the same application can be run – and that’s not even mentioning the myriad existing ways to handle static content. Usually, the approach boils down to one of the following:

  • A single script handles incoming requests for all URIs, routing them internally
  • Each request is handled by the script explicitly named in the URI

NGINX Unit has included options for both approaches since the very introduction of PHP support. Nonetheless, the sheer number of possible combinations in real deployments was begging for a change. To give you a picture: almost every time a PHP application was set up in NGINX Unit, several application objects had to be defined, along with intricate routing rules that channeled the requests among them, as in this example:

{
“routes”: [
{
“match”: {
“uri”: [
.php”, “.php/*”
]
},

        "action": {
            "pass": "applications/direct"
        }
    },

    {
        "action": {
            "pass": "applications/script_index"
        }
    },
],

"applications": {
    "direct": {
        "type": "php",
        "processes": {
            "max": 5,
            "spare": 0
        },
        "user": "www-data",
        "group": "www-data",
        "root": "/var/www/app/",
        "index": "index.php"
    },

    "script_index": {
        "type": "php",
        "processes": {
            "max": 20,
            "spare": 5
        },
        "user": "www-data",
        "group": "www-data",
        "root": "/var/www/app/",
        "script": "index.php"
    }
}

}

view rawphp_config_without_targets hosted with ❤ by GitHub

Here, requests that explicitly name a PHP script are passed to the direct app, while all other requests are passed to the script_index app, which handles them by means of the multipurpose index.php script at the root of the app (pretty URIs are a common use case for the latter). Because the two apps are distinct, they must be separately configured in the applications object, but you can readily see there are quite a few duplicated options.

Such redundancy makes sense if you wish to fine‑tune the settings for each section of your web application (section being loosely defined as an arbitrary set of PHP scripts and URIs that need to be configured as a single entity) or assign a dedicated process pool to each individual section of your app. However, NGINX Unit runs such sections as separate apps even when that is not your end goal, which can result in unwanted overhead: the app requires several processes to run (at least one per each section), but some of them remain idle most of the time (wasting CPU cycles and memory), while others may be constantly overloaded with a torrent of requests. There needs to be a way to share processes between the individual sections of an app.

Moreover, the one-app-per-section approach requires repeating the same setting values over and over, causing redundancy, unnecessary risk of misconfiguration, and lack of logical cohesion between the individual sections of the app that were represented as standalone entities. To update a redundant setting, multiple non‑atomic configuration updates (API calls) are necessary, which raises concerns about potential inconsistencies. All of these problems led us to introduce a new approach in version 1.18.0.

Now, instead of having to set up a single application object for each and every section of your app that handles its scripts in a customized way, you can use the new targets option to make them all peacefully work together under the roof of a single app:

{
“routes”: [
{
“match”: {
“uri”: [
.php”, “.php/*”
]
},

        "action": {
            "pass": "applications/phpapp/direct"
        }
    },

    {
        "action": {
            "pass": "applications/phpapp/script_index"
        }
    },
],

"applications": {
    "phpapp": {
        "type": "php",
        "user": "www-data",
        "group": "www-data",
        "targets": {
            "direct": {
                "root": "/var/www/app/"
            },

            "script_index": {
                "root": "/var/www/app/",
                "script": "index.php"
            }
        }
    }
}

}

view rawphp_config_with_targets hosted with ❤ by GitHub

Each target within the app can have its own combination of the rootindex, and script options, independently running its own scripts. However, all targets share such application‑wide settings as isolationlimitsoptions, and processes. Keep in mind, however, that it’s not currently possible to manage resource distribution between individual targets within an app.

The new approach centralizes and error‑proofs application management while allowing you to flexibly control the many endpoints of any application with individual targets. Moreover, you can combine it with the older method if need be, grouping or decoupling your app configurations as you see fit.

For further details and examples, see our Targets configuration guide and how‑tos for PHP‑based apps.

URL Encoding

NGINX Plus 1.18.0 implements URL encoding (also called percent encoding), which is used in URIs most often to represent characters outside the English‑language alphabet (such as characters with diacritical markings like the umlaut or tilde) and characters that can have a special meaning (such as the forward slash as the divider between elements in a URI) when they don’t have that meaning. You can use URL encoding to escape characters in the passarguments, and uri options. There are two main use cases.

First, you can escape the forward‑slash character as %2F in the argument to a pass option:

{
“listeners”: {
“*:80”: {
“pass”: “routes/slashes%2Fin%2Froute%2Fname”
}
},

"routes": {
     "slashes/in/route/name": [
     ]
}

}

view rawescape_forward_slash hosted with ❤ by GitHub

Second, you can create filters with the uri and arguments options that contain characters which have special meaning in NGINX Unit routing, or even target individual bytes:

{
“routes”: {
“slashes/in/route/name”: [
{
“match”: {
“uri”: “/%2A”,
“arguments”: {
“%25”: “%21%C3*”
}
},

            "action": {
                "return": 301,
                "location": "http://fancyurls.example.com"
            }
        }
    ]
}

}

view rawfancy_urls hosted with ❤ by GitHub

Here, the uri filter expects a single literal asterisk (*, URL‑encoded as %2A), and the arguments filter expects the percent sign (%, URL‑encoded as %25) as the key with a value consisting of an exclamation point (!, URL‑encoded as %21) followed by a diacritic UTF‑8 character such as Ö or Å: the byte value %C3 usually signals a set of diacritic characters and is followed here by an arbitrary byte sequence (represented by the asterisk in the value definition).

These two sample queries illustrate the effect of the complete configuration:

$ curl -v 'http://localhost:80/*?%=!Ü'
...
< HTTP/1.1 301 Moved Permanently
< Location: http://fancyurls.example.com
< Server: Unit/1.18.0

$ curl -v 'http://localhost:80/*?%=!ò'
...
< HTTP/1.1 301 Moved Permanently
< Location: http://fancyurls.example.com
< Server: Unit/1.18.0

You can see that both requests are successfully redirected which means NGINX Unit has matched the asterisk and exclamation point literals and the diacritic characters as intended. Effectively, this means you can single out arbitrary byte sequences and characters in different encodings.

Enhancements Introduced in NGINX Unit 1.17.0

In addition to the headliner changes we made in NGINX Unit 1.18.0, version 1.17.0 introduced some notable but smaller‑scale updates.

Instant Responses and Redirects

NGINX Unit 1.17.0 introduced support for instant responses and redirections during routing. The new route action called return allows you to reply to a request with an arbitrary HTTP response status code and also provide a redirect location when that is required by the status code’s semantics. Here we define a permanent redirect for the /legacy/ URI:

{
“match”: {
“uri”: “/legacy/*”
},

"action": {
    "return": 301,
    "location": "http://legacy.example.com"
}

}

view rawredirect hosted with ❤ by GitHub

Fractional Server Weights in Upstreams

Version 1.17.0 also added support for fractional weights in the upstreams object introduced in version 1.16.0. With the original integer weights, you often had to adjust the entire weighting scheme to change a single server’s share of requests. Consider this initial configuration, which divides requests evenly among three servers (each has the default weight of 1):

{
“servers”: {
“192.168.0.100:8080”,
“192.168.0.101:8080”,
“192.168.0.102:8080”
}
}

view rawequal_weights hosted with ❤ by GitHub

If you then want one server to get only half as many requests as the others (a distribution represented with integers as 2:2:1), you actually have to change the weight on the two servers whose relative proportion is staying the same, which is somewhat counterintuitive:

{
“servers”: {
“192.168.0.100:8080”: {
“weight”: 2,
},
“192.168.0.101:8080”: {
“weight”: 2,
},
“192.168.0.102:8080”
}
}

view raw2_2_1_weights hosted with ❤ by GitHub

With fractional weights, you need to change the weight only on the server whose proportion is changing:

{
“servers”: {
“192.168.0.100:8080”,
“192.168.0.101:8080”,
“192.168.0.102:8080”: {
“weight”: 0.5
}
}
}

view rawfractional_weights hosted with ❤ by GitHub

The difference between the two schemes is not great with just three servers, but fractional weights can significantly reduce the number of necessary changes when there are many servers.

Conclusion

This is a brief recap of the new features in versions 1.17.0 and 1.18.0. As mentioned above, very soon we’ll publish a dedicated post delving into the details of the newly introduced rootfs feature: its technicalities and advantages, potential use cases, less-than-obvious risks, and unexpected side effects. Stay tuned!

Meanwhile, the NGINX Unit team continues to work on even more fabulous improvements while simultaneously perfecting what’s already there: our current roadmap includes plans to support configuration variables, add some HTTP goodness, and extend wildcard support in request matching patterns.

For a list of the changes and bug fixes in releases 1.17.0 and 1.18.0, see the NGINX Unit changelog.

NGINX Plus subscribers get support for NGINX Unit at no additional charge. Start a free 30‑day trial of NGINX Plus today.

Source: nginx

Author

oDesk Software

Leave a comment