Blame modules/mappers/mod_vhost_alias.c

Packit 90a5c9
/* Licensed to the Apache Software Foundation (ASF) under one or more
Packit 90a5c9
 * contributor license agreements.  See the NOTICE file distributed with
Packit 90a5c9
 * this work for additional information regarding copyright ownership.
Packit 90a5c9
 * The ASF licenses this file to You under the Apache License, Version 2.0
Packit 90a5c9
 * (the "License"); you may not use this file except in compliance with
Packit 90a5c9
 * the License.  You may obtain a copy of the License at
Packit 90a5c9
 *
Packit 90a5c9
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 90a5c9
 *
Packit 90a5c9
 * Unless required by applicable law or agreed to in writing, software
Packit 90a5c9
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 90a5c9
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 90a5c9
 * See the License for the specific language governing permissions and
Packit 90a5c9
 * limitations under the License.
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * mod_vhost_alias.c: support for dynamically configured mass virtual hosting
Packit 90a5c9
 *
Packit 90a5c9
 * Copyright (c) 1998-1999 Demon Internet Ltd.
Packit 90a5c9
 *
Packit 90a5c9
 * This software was submitted by Demon Internet to the Apache Software Foundation
Packit 90a5c9
 * in May 1999. Future revisions and derivatives of this source code
Packit 90a5c9
 * must acknowledge Demon Internet as the original contributor of
Packit 90a5c9
 * this module. All other licensing and usage conditions are those
Packit 90a5c9
 * of the Apache Software Foundation.
Packit 90a5c9
 *
Packit 90a5c9
 * Originally written by Tony Finch <fanf@demon.net> <dot@dotat.at>.
Packit 90a5c9
 *
Packit 90a5c9
 * Implementation ideas were taken from mod_alias.c. The overall
Packit 90a5c9
 * concept is derived from the OVERRIDE_DOC_ROOT/OVERRIDE_CGIDIR
Packit 90a5c9
 * patch to Apache 1.3b3 and a similar feature in Demon's thttpd,
Packit 90a5c9
 * both written by James Grinter <jrg@blodwen.demon.co.uk>.
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
#include "apr.h"
Packit 90a5c9
#include "apr_strings.h"
Packit 90a5c9
#include "ap_hooks.h"
Packit 90a5c9
#include "apr_lib.h"
Packit 90a5c9
Packit 90a5c9
#define APR_WANT_STRFUNC
Packit 90a5c9
#include "apr_want.h"
Packit 90a5c9
Packit 90a5c9
#include "httpd.h"
Packit 90a5c9
#include "http_config.h"
Packit 90a5c9
#include "http_core.h"
Packit 90a5c9
#include "http_request.h"  /* for ap_hook_translate_name */
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
module AP_MODULE_DECLARE_DATA vhost_alias_module;
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * basic configuration things
Packit 90a5c9
 * we abbreviate "mod_vhost_alias" to "mva" for shorter names
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
typedef enum {
Packit 90a5c9
    VHOST_ALIAS_UNSET, VHOST_ALIAS_NONE, VHOST_ALIAS_NAME, VHOST_ALIAS_IP
Packit 90a5c9
} mva_mode_e;
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * Per-server module config record.
Packit 90a5c9
 */
Packit 90a5c9
typedef struct mva_sconf_t {
Packit 90a5c9
    const char *doc_root;
Packit 90a5c9
    const char *cgi_root;
Packit 90a5c9
    mva_mode_e doc_root_mode;
Packit 90a5c9
    mva_mode_e cgi_root_mode;
Packit 90a5c9
} mva_sconf_t;
Packit 90a5c9
Packit 90a5c9
static void *mva_create_server_config(apr_pool_t *p, server_rec *s)
Packit 90a5c9
{
Packit 90a5c9
    mva_sconf_t *conf;
Packit 90a5c9
Packit 90a5c9
    conf = (mva_sconf_t *) apr_pcalloc(p, sizeof(mva_sconf_t));
Packit 90a5c9
    conf->doc_root = NULL;
Packit 90a5c9
    conf->cgi_root = NULL;
Packit 90a5c9
    conf->doc_root_mode = VHOST_ALIAS_UNSET;
Packit 90a5c9
    conf->cgi_root_mode = VHOST_ALIAS_UNSET;
Packit 90a5c9
    return conf;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void *mva_merge_server_config(apr_pool_t *p, void *parentv, void *childv)
Packit 90a5c9
{
Packit 90a5c9
    mva_sconf_t *parent = (mva_sconf_t *) parentv;
Packit 90a5c9
    mva_sconf_t *child = (mva_sconf_t *) childv;
Packit 90a5c9
    mva_sconf_t *conf;
Packit 90a5c9
Packit 90a5c9
    conf = (mva_sconf_t *) apr_pcalloc(p, sizeof(*conf));
Packit 90a5c9
    if (child->doc_root_mode == VHOST_ALIAS_UNSET) {
Packit 90a5c9
        conf->doc_root_mode = parent->doc_root_mode;
Packit 90a5c9
        conf->doc_root = parent->doc_root;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        conf->doc_root_mode = child->doc_root_mode;
Packit 90a5c9
        conf->doc_root = child->doc_root;
Packit 90a5c9
    }
Packit 90a5c9
    if (child->cgi_root_mode == VHOST_ALIAS_UNSET) {
Packit 90a5c9
        conf->cgi_root_mode = parent->cgi_root_mode;
Packit 90a5c9
        conf->cgi_root = parent->cgi_root;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        conf->cgi_root_mode = child->cgi_root_mode;
Packit 90a5c9
        conf->cgi_root = child->cgi_root;
Packit 90a5c9
    }
Packit 90a5c9
    return conf;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * These are just here to tell us what vhost_alias_set should do.
Packit 90a5c9
 * We don't put anything into them; we just use the cell addresses.
Packit 90a5c9
 */
Packit 90a5c9
static int vhost_alias_set_doc_root_ip,
Packit 90a5c9
    vhost_alias_set_cgi_root_ip,
Packit 90a5c9
    vhost_alias_set_doc_root_name,
Packit 90a5c9
    vhost_alias_set_cgi_root_name;
Packit 90a5c9
Packit 90a5c9
static const char *vhost_alias_set(cmd_parms *cmd, void *dummy, const char *map)
Packit 90a5c9
{
Packit 90a5c9
    mva_sconf_t *conf;
Packit 90a5c9
    mva_mode_e mode, *pmode;
Packit 90a5c9
    const char **pmap;
Packit 90a5c9
    const char *p;
Packit 90a5c9
Packit 90a5c9
    conf = (mva_sconf_t *) ap_get_module_config(cmd->server->module_config,
Packit 90a5c9
                                                &vhost_alias_module);
Packit 90a5c9
    /* there ought to be a better way of doing this */
Packit 90a5c9
    if (&vhost_alias_set_doc_root_ip == cmd->info) {
Packit 90a5c9
        mode = VHOST_ALIAS_IP;
Packit 90a5c9
        pmap = &conf->doc_root;
Packit 90a5c9
        pmode = &conf->doc_root_mode;
Packit 90a5c9
    }
Packit 90a5c9
    else if (&vhost_alias_set_cgi_root_ip == cmd->info) {
Packit 90a5c9
        mode = VHOST_ALIAS_IP;
Packit 90a5c9
        pmap = &conf->cgi_root;
Packit 90a5c9
        pmode = &conf->cgi_root_mode;
Packit 90a5c9
    }
Packit 90a5c9
    else if (&vhost_alias_set_doc_root_name == cmd->info) {
Packit 90a5c9
        mode = VHOST_ALIAS_NAME;
Packit 90a5c9
        pmap = &conf->doc_root;
Packit 90a5c9
        pmode = &conf->doc_root_mode;
Packit 90a5c9
    }
Packit 90a5c9
    else if (&vhost_alias_set_cgi_root_name == cmd->info) {
Packit 90a5c9
        mode = VHOST_ALIAS_NAME;
Packit 90a5c9
        pmap = &conf->cgi_root;
Packit 90a5c9
        pmode = &conf->cgi_root_mode;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return "INTERNAL ERROR: unknown command info";
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    if (!ap_os_is_path_absolute(cmd->pool, map)) {
Packit 90a5c9
        if (strcasecmp(map, "none")) {
Packit 90a5c9
            return "format string must be an absolute path, or 'none'";
Packit 90a5c9
        }
Packit 90a5c9
        *pmap = NULL;
Packit 90a5c9
        *pmode = VHOST_ALIAS_NONE;
Packit 90a5c9
        return NULL;
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    /* sanity check */
Packit 90a5c9
    p = map;
Packit 90a5c9
    while (*p != '\0') {
Packit 90a5c9
        if (*p++ != '%') {
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        /* we just found a '%' */
Packit 90a5c9
        if (*p == 'p' || *p == '%') {
Packit 90a5c9
            ++p;
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        /* optional dash */
Packit 90a5c9
        if (*p == '-') {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
        /* digit N */
Packit 90a5c9
        if (apr_isdigit(*p)) {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
        else {
Packit 90a5c9
            return "syntax error in format string";
Packit 90a5c9
        }
Packit 90a5c9
        /* optional plus */
Packit 90a5c9
        if (*p == '+') {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
        /* do we end here? */
Packit 90a5c9
        if (*p != '.') {
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        ++p;
Packit 90a5c9
        /* optional dash */
Packit 90a5c9
        if (*p == '-') {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
        /* digit M */
Packit 90a5c9
        if (apr_isdigit(*p)) {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
        else {
Packit 90a5c9
            return "syntax error in format string";
Packit 90a5c9
        }
Packit 90a5c9
        /* optional plus */
Packit 90a5c9
        if (*p == '+') {
Packit 90a5c9
            ++p;
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    *pmap = map;
Packit 90a5c9
    *pmode = mode;
Packit 90a5c9
    return NULL;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static const command_rec mva_commands[] =
Packit 90a5c9
{
Packit 90a5c9
    AP_INIT_TAKE1("VirtualScriptAlias", vhost_alias_set,
Packit 90a5c9
                  &vhost_alias_set_cgi_root_name, RSRC_CONF,
Packit 90a5c9
                  "how to create a ScriptAlias based on the host"),
Packit 90a5c9
    AP_INIT_TAKE1("VirtualDocumentRoot", vhost_alias_set,
Packit 90a5c9
                  &vhost_alias_set_doc_root_name, RSRC_CONF,
Packit 90a5c9
                  "how to create the DocumentRoot based on the host"),
Packit 90a5c9
    AP_INIT_TAKE1("VirtualScriptAliasIP", vhost_alias_set,
Packit 90a5c9
                  &vhost_alias_set_cgi_root_ip, RSRC_CONF,
Packit 90a5c9
                  "how to create a ScriptAlias based on the host"),
Packit 90a5c9
    AP_INIT_TAKE1("VirtualDocumentRootIP", vhost_alias_set,
Packit 90a5c9
                  &vhost_alias_set_doc_root_ip, RSRC_CONF,
Packit 90a5c9
                  "how to create the DocumentRoot based on the host"),
Packit 90a5c9
    { NULL }
Packit 90a5c9
};
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * This really wants to be a nested function
Packit 90a5c9
 * but C is too feeble to support them.
Packit 90a5c9
 */
Packit 90a5c9
static APR_INLINE void vhost_alias_checkspace(request_rec *r, char *buf,
Packit 90a5c9
                                             char **pdest, int size)
Packit 90a5c9
{
Packit 90a5c9
    /* XXX: what if size > HUGE_STRING_LEN? */
Packit 90a5c9
    if (*pdest + size > buf + HUGE_STRING_LEN) {
Packit 90a5c9
        **pdest = '\0';
Packit 90a5c9
        if (r->filename) {
Packit 90a5c9
            r->filename = apr_pstrcat(r->pool, r->filename, buf, NULL);
Packit 90a5c9
        }
Packit 90a5c9
        else {
Packit 90a5c9
            r->filename = apr_pstrdup(r->pool, buf);
Packit 90a5c9
        }
Packit 90a5c9
        *pdest = buf;
Packit 90a5c9
    }
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void vhost_alias_interpolate(request_rec *r, const char *name,
Packit 90a5c9
                                    const char *map, const char *uri)
Packit 90a5c9
{
Packit 90a5c9
    /* 0..9 9..0 */
Packit 90a5c9
    enum { MAXDOTS = 19 };
Packit 90a5c9
    const char *dots[MAXDOTS+1];
Packit 90a5c9
    int ndots;
Packit 90a5c9
Packit 90a5c9
    char buf[HUGE_STRING_LEN];
Packit 90a5c9
    char *dest;
Packit 90a5c9
    const char *docroot;
Packit 90a5c9
Packit 90a5c9
    int N, M, Np, Mp, Nd, Md;
Packit 90a5c9
    const char *start, *end;
Packit 90a5c9
Packit 90a5c9
    const char *p;
Packit 90a5c9
Packit 90a5c9
    ndots = 0;
Packit 90a5c9
    dots[ndots++] = name-1; /* slightly naughty */
Packit 90a5c9
    for (p = name; *p; ++p) {
Packit 90a5c9
        if (*p == '.' && ndots < MAXDOTS) {
Packit 90a5c9
            dots[ndots++] = p;
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    dots[ndots] = p;
Packit 90a5c9
Packit 90a5c9
    r->filename = NULL;
Packit 90a5c9
Packit 90a5c9
    dest = buf;
Packit 90a5c9
    while (*map) {
Packit 90a5c9
        if (*map != '%') {
Packit 90a5c9
            /* normal characters */
Packit 90a5c9
            vhost_alias_checkspace(r, buf, &dest, 1);
Packit 90a5c9
            *dest++ = *map++;
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        /* we are in a format specifier */
Packit 90a5c9
        ++map;
Packit 90a5c9
        /* %% -> % */
Packit 90a5c9
        if (*map == '%') {
Packit 90a5c9
            ++map;
Packit 90a5c9
            vhost_alias_checkspace(r, buf, &dest, 1);
Packit 90a5c9
            *dest++ = '%';
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        /* port number */
Packit 90a5c9
        if (*map == 'p') {
Packit 90a5c9
            ++map;
Packit 90a5c9
            /* no. of decimal digits in a short plus one */
Packit 90a5c9
            vhost_alias_checkspace(r, buf, &dest, 7);
Packit 90a5c9
            dest += apr_snprintf(dest, 7, "%d", ap_get_server_port(r));
Packit 90a5c9
            continue;
Packit 90a5c9
        }
Packit 90a5c9
        /* deal with %-N+.-M+ -- syntax is already checked */
Packit 90a5c9
        M = 0;   /* value */
Packit 90a5c9
        Np = Mp = 0; /* is there a plus? */
Packit 90a5c9
        Nd = Md = 0; /* is there a dash? */
Packit 90a5c9
        if (*map == '-') ++map, Nd = 1;
Packit 90a5c9
        N = *map++ - '0';
Packit 90a5c9
        if (*map == '+') ++map, Np = 1;
Packit 90a5c9
        if (*map == '.') {
Packit 90a5c9
            ++map;
Packit 90a5c9
            if (*map == '-') {
Packit 90a5c9
                ++map, Md = 1;
Packit 90a5c9
            }
Packit 90a5c9
            M = *map++ - '0';
Packit 90a5c9
            if (*map == '+') {
Packit 90a5c9
                ++map, Mp = 1;
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
        /* note that N and M are one-based indices, not zero-based */
Packit 90a5c9
        start = dots[0]+1; /* ptr to the first character */
Packit 90a5c9
        end = dots[ndots]; /* ptr to the character after the last one */
Packit 90a5c9
        if (N != 0) {
Packit 90a5c9
            if (N > ndots) {
Packit 90a5c9
                start = "_";
Packit 90a5c9
                end = start+1;
Packit 90a5c9
            }
Packit 90a5c9
            else if (!Nd) {
Packit 90a5c9
                start = dots[N-1]+1;
Packit 90a5c9
                if (!Np) {
Packit 90a5c9
                    end = dots[N];
Packit 90a5c9
                }
Packit 90a5c9
            }
Packit 90a5c9
            else {
Packit 90a5c9
                if (!Np) {
Packit 90a5c9
                    start = dots[ndots-N]+1;
Packit 90a5c9
                }
Packit 90a5c9
                end = dots[ndots-N+1];
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
        if (M != 0) {
Packit 90a5c9
            if (M > end - start) {
Packit 90a5c9
                start = "_";
Packit 90a5c9
                end = start+1;
Packit 90a5c9
            }
Packit 90a5c9
            else if (!Md) {
Packit 90a5c9
                start = start+M-1;
Packit 90a5c9
                if (!Mp) {
Packit 90a5c9
                    end = start+1;
Packit 90a5c9
                }
Packit 90a5c9
            }
Packit 90a5c9
            else {
Packit 90a5c9
                if (!Mp) {
Packit 90a5c9
                    start = end-M;
Packit 90a5c9
                }
Packit 90a5c9
                end = end-M+1;
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
        vhost_alias_checkspace(r, buf, &dest, end - start);
Packit 90a5c9
        for (p = start; p < end; ++p) {
Packit 90a5c9
            *dest++ = apr_tolower(*p);
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    /* no double slashes */
Packit 90a5c9
    if (dest - buf > 0 && dest[-1] == '/') {
Packit 90a5c9
        --dest;
Packit 90a5c9
    }
Packit 90a5c9
    *dest = '\0';
Packit 90a5c9
Packit 90a5c9
    if (r->filename)
Packit 90a5c9
        docroot = apr_pstrcat(r->pool, r->filename, buf, NULL);
Packit 90a5c9
    else
Packit 90a5c9
        docroot = apr_pstrdup(r->pool, buf);
Packit 90a5c9
    r->filename = apr_pstrcat(r->pool, docroot, uri, NULL);
Packit 90a5c9
    ap_set_context_info(r, NULL, docroot);
Packit 90a5c9
    ap_set_document_root(r, docroot);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static int mva_translate(request_rec *r)
Packit 90a5c9
{
Packit 90a5c9
    mva_sconf_t *conf;
Packit 90a5c9
    const char *name, *map, *uri;
Packit 90a5c9
    mva_mode_e mode;
Packit 90a5c9
    const char *cgi;
Packit 90a5c9
Packit 90a5c9
    conf = (mva_sconf_t *) ap_get_module_config(r->server->module_config,
Packit 90a5c9
                                              &vhost_alias_module);
Packit 90a5c9
    cgi = NULL;
Packit 90a5c9
    if (conf->cgi_root) {
Packit 90a5c9
        cgi = strstr(r->uri, "cgi-bin/");
Packit 90a5c9
        if (cgi && (cgi != r->uri + strspn(r->uri, "/"))) {
Packit 90a5c9
            cgi = NULL;
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    if (cgi) {
Packit 90a5c9
        mode = conf->cgi_root_mode;
Packit 90a5c9
        map = conf->cgi_root;
Packit 90a5c9
        uri = cgi + strlen("cgi-bin");
Packit 90a5c9
    }
Packit 90a5c9
    else if (r->uri[0] == '/') {
Packit 90a5c9
        mode = conf->doc_root_mode;
Packit 90a5c9
        map = conf->doc_root;
Packit 90a5c9
        uri = r->uri;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return DECLINED;
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    if (mode == VHOST_ALIAS_NAME) {
Packit 90a5c9
        name = ap_get_server_name(r);
Packit 90a5c9
    }
Packit 90a5c9
    else if (mode == VHOST_ALIAS_IP) {
Packit 90a5c9
        name = r->connection->local_ip;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return DECLINED;
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    /* ### There is an optimization available here to determine the
Packit 90a5c9
     * absolute portion of the path from the server config phase,
Packit 90a5c9
     * through the first % segment, and note that portion of the path
Packit 90a5c9
     * canonical_path buffer.
Packit 90a5c9
     */
Packit 90a5c9
    r->canonical_filename = "";
Packit 90a5c9
    vhost_alias_interpolate(r, name, map, uri);
Packit 90a5c9
Packit 90a5c9
    if (cgi) {
Packit 90a5c9
        /* see is_scriptaliased() in mod_cgi */
Packit 90a5c9
        r->handler = "cgi-script";
Packit 90a5c9
        apr_table_setn(r->notes, "alias-forced-type", r->handler);
Packit 90a5c9
        ap_set_context_info(r, "/cgi-bin", NULL);
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    return OK;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void register_hooks(apr_pool_t *p)
Packit 90a5c9
{
Packit 90a5c9
    static const char * const aszPre[]={ "mod_alias.c","mod_userdir.c",NULL };
Packit 90a5c9
Packit 90a5c9
    ap_hook_translate_name(mva_translate, aszPre, NULL, APR_HOOK_MIDDLE);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
AP_DECLARE_MODULE(vhost_alias) =
Packit 90a5c9
{
Packit 90a5c9
    STANDARD20_MODULE_STUFF,
Packit 90a5c9
    NULL,                       /* dir config creater */
Packit 90a5c9
    NULL,                       /* dir merger --- default is to override */
Packit 90a5c9
    mva_create_server_config,   /* server config */
Packit 90a5c9
    mva_merge_server_config,    /* merge server configs */
Packit 90a5c9
    mva_commands,               /* command apr_table_t */
Packit 90a5c9
    register_hooks              /* register hooks */
Packit 90a5c9
};
Packit 90a5c9