Как использовать "И", "ИЛИ" для RewriteCond на Apache?

Это как использовать AND, OR для RewriteCond в Apache?

rewritecond A [or]
rewritecond B
rewritecond C [or]
rewritecond D
RewriteRule ... something

становится if ( (A or B) and (C or D) ) rewrite_it.

Итак, похоже, что "ИЛИ" имеет более высокий приоритет, чем "И"? Есть ли способ легко сказать, как в синтаксисе (A or B) and (C or D)?

Ответ 1

Это интересный вопрос, и поскольку он не очень подробно объясняется в документации, я отвечу на это, пройдя исходный код mod_rewrite; демонстрирует большое преимущество open-source.

В верхней части вы быстро обнаружите define, используемые для обозначения этих флагов:

#define CONDFLAG_NONE               1<<0
#define CONDFLAG_NOCASE             1<<1
#define CONDFLAG_NOTMATCH           1<<2
#define CONDFLAG_ORNEXT             1<<3
#define CONDFLAG_NOVARY             1<<4

и поиск CONDFLAG_ORNEXT подтверждает, что он используется на основе существования флага [OR]:

else if (   strcasecmp(key, "ornext") == 0
         || strcasecmp(key, "OR") == 0    ) {
    cfg->flags |= CONDFLAG_ORNEXT;
}

Следующее вхождение флага - это фактическая реализация где вы найдете цикл, который проходит через все RewriteConditions, RewriteRule, и что он в основном делает (разделение, комментарии добавлены для ясности):

# loop through all Conditions that precede this Rule
for (i = 0; i < rewriteconds->nelts; ++i) {
    rewritecond_entry *c = &conds[i];

    # execute the current Condition, see if it matches
    rc = apply_rewrite_cond(c, ctx);

    # does this Condition have an 'OR' flag?
    if (c->flags & CONDFLAG_ORNEXT) {
        if (!rc) {
            /* One condition is false, but another can be still true. */
            continue;
        }
        else {
            /* skip the rest of the chained OR conditions */
            while (   i < rewriteconds->nelts
                   && c->flags & CONDFLAG_ORNEXT) {
                c = &conds[++i];
            }
        }
    }
    else if (!rc) {
        return 0;
    }
}

Вы должны уметь это интерпретировать; это означает, что OR имеет более высокий приоритет, и ваш пример действительно приводит к if ( (A OR B) AND (C OR D) ). Если бы у вас были, например, следующие условия:

RewriteCond A [or]
RewriteCond B [or]
RewriteCond C
RewriteCond D

он будет интерпретироваться как if ( (A OR B OR C) and D ).

Ответ 2

После многих трудностей и достижения общего, гибкого и более удобочитаемого решения, в моем случае я закончил тем, что сохранил результаты ИЛИ в переменные ENV и выполнил И этих переменных.

# RESULT_ONE = A OR B
RewriteRule ^ - [E=RESULT_ONE:False]
RewriteCond ...A... [OR]
RewriteCond ...B...
RewriteRule ^ - [E=RESULT_ONE:True]

# RESULT_TWO = C OR D
RewriteRule ^ - [E=RESULT_TWO:False]
RewriteCond ...C... [OR]
RewriteCond ...D...
RewriteRule ^ - [E=RESULT_TWO:True]

# if ( RESULT_ONE AND RESULT_TWO ) then ( RewriteRule ...something... )
RewriteCond %{ENV:RESULT_ONE} =True
RewriteCond %{ENV:RESULT_TWO} =True
RewriteRule ...something...

Требования: