Guardiamo in faccia la realtà, puoi offrire e vendere il servizio migliore del mondo ma (spesso) il cliente/utente medio lo giudicherà per la “copertina”: se non si presenta bene fa schifo, se è stilosa allora forse apprezzerà anche il resto.

Hai voglia a spiegare che si tratta di sicurezza, che è importante avere i servizi UTM attivi (VERO?!?), che il prodotto è buono, se poi lasci un banale Deny Message a default grigio smorto. watchguard http-proxy deny message

E poi diciamocela tutta, oltre che bruttina è pure poco funzionale: la WebBlocker Category non sempre ci azzecca oppure ritorna Uncategorized per il sito della zia Pina con cui l’utente fa business, e quindi via di telefonate o mail per richiedere lo sblocco. In alternativa puoi abilitare il WebBlocker Override ma per quali categorie o per quali utenti? Ti vai a fidare del buonsenso dell’utente?! […] Ecco appunto!

Che te ne pare invece di personalizzare ed arricchire il Deny Message ad esempio così? watchguard http-proxy deny message custom

Liberiamo la fantasia

Che ci vuole? Prendo il primo template HTML che trovo online, un ritocchino al CSS, aggiungo un po’ di JavaScript che fa figo e BAM!

Non funziona…

Nessuna specifica dettagliata ma, per quanto provato in lab, o c’è un limite sulla lunghezza del codice, o qualcosa sul parsing interno o altro che non ho intenzione di approfondire. Fatto sta che il codice restituito al browser viene troncato.

Quindi?

1
<meta http-equiv="refresh" content="0; url=https://mioserver.contoso.local/customdeny.html?url=%(url-host)%&reason=%(reason)%&user=%(user-name)%">

BAM!

Adesso sì! Nel nostro Deny Message di default abbiamo aggiunto un bel redirect all’interno del tag <head></head> che punti ad una pagina customdeny.html sul server web interno mioserver.contoso.local e passiamo tutte le variabili che ci interessano come query strings.

Customizziamo la Deny Page - le basi

Tralasciando la veste grafica, per prima cosa definiamo le variabili da mostrare all’utente in customdeny.html.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
<script>
	var params = new URLSearchParams(window.location.search);
	var url = params.get("url");
	var reason = params.get("reason");
	var user = params.get("user");

	document.addEventListener("DOMContentLoaded", function () {
		document.getElementById("reason").appendChild(document.createTextNode(reason));
		document.getElementById("user").appendChild(document.createTextNode(user));
		document.getElementById("host").appendChild(document.createTextNode(url));
	});
</script>
<div>
	<p id="host"><b>Sito: </b></p>
	<p id="reason"><b>Motivo: </b></p>
	<p id="user"><b>Utente: </b></p>
</div>

Queste definite sono quelle fondamentali ma volendo potremmo aggiungere anche le altre che il nostro Firebox supporta:

  • %(transaction)% - Includes Request or Response to show which side of the transaction caused the packet to be denied
  • %(method)% - Includes the request method from the request
  • %(url-path)% - Includes the path component of the URL
  • %(serial)% - Includes the serial number of the Firebox
  • %(firewall)% - Includes the Firebox name

Ok, tutto molto bello ma quel mirabolante RICHIEDI SBLOCCO a portata di click che c’era prima?

Deny Page con richiesta sblocco e invio mail

Aggiungiamo il destinatario della mail e due distinte funzioni JavaScript: requireUnlock e sendUnlock, perchè?

Nella prima costringeremo l’utente a motivare la sua richiesta perchè è scientificamente provato che solo pochi tra i più temerari proveranno a farsi sbloccare siti a caso scrivendo nero su bianco che gli serve per lavoro, nella seconda funzione invece daremo il via alla mail con annessa motivazione.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
var recipient = "[email protected]";
function requireUnlock() {
	let req = prompt("Motivo della richiesta:", "");
	if (req == null || req.trim().length < 5) {
		alert("Necessario inserire una motivazione valida per la richiesta");
	}
	else {
		sendUnlock(req);
	}
}

function sendUnlock(unlockReason) {
	var subject = "Richiesta sblocco sito web";
	var row1 = "Utente:\t" + user;
	var row2 = "Sito:\t" + url;
	var row3 = "Blocco:\t" + reason;
	var row4 = "Motivo:\t" + unlockReason;
	var message = row1 + "\n" + row2 + "\n" + row3 + "\n" + row4;
	var mailData = encodeURIComponent(message);
	window.location.href = "mailto:" + recipient + "?subject=" + subject + "&body=" + mailData;
}
1
<button onclick="requireUnlock()">RICHIEDI SBLOCCO</button>

Ed ecco il risultato, se l’utente motiva correttamente la richiesta verrà creata una mail precompilata da inviare con il mail client di default. watchguard http-proxy deny message richiedi sblocco watchguard http-proxy deny message sblocco notifica mail

In caso contrario, motivazione vuota o lunghezza < di 5 caratteri mostriamo un messaggio di errore. watchguard http-proxy deny message accesso negato

Non male no? Non ti resta che personalizzare AKA brandizzare il tuo customdeny.html e se ti piace il copia/incolla facile trovi il codice completo a fine articolo.

Variazioni

Se non vuoi/puoi usare un server web interno, fai qualche tentativo inserendo le parti JavaScript rilevanti direttamente nel Deny Message del Firebox, in questo caso le variabili JavaScript andranno definite così:

1
2
3
var url = "%(url-host)%";
var reason = "%(reason)%";
var user = "%(user-name)%";

Occhio però alla lunghezza totale del codice e scordati le immagini in base64 che vedi nell’esempio, usa una <img/> src che sia raggiungibile da tutti o evita proprio le immagini.

Altra possibilità da tenere presente, con un minimo di conoscenze di programmazione, è quella di usare la funzione sendUnlock per chiamare un Web Service, magari anche con autenticazione SSO, che si occupi di mandare la mail senza dover passare dall’Outlook dell’utente… se fossi “studiato” ti direi che ne guadagnarebbe la User Experience ;)

Il copia/incolla definitivo

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
<!DOCTYPE html>
<html lang="en" xmlns="http://www.w3.org/1999/xhtml">
<head>
    <meta charset="utf-8" />
    <title>Access Denied</title>
    <style type="text/css">
        html, body {
            height: 100%;
            margin: 0;
            font: normal 1em Arial;
            color: #666;
            background: #fff;
        }

        h1 {
            font: bold 3.5em Arial;
            font-style: italic;
            color: #373737;
        }

        h2 {
            font: bold 2em Tahoma;
            color: #d61e1e;
        }

        .outer {
            padding-top: 10%;
            margin: 0 auto;
            text-align: center;
            width:50%;
        }

        .inner {
            border: solid 10px #4c4c4c;
            padding: 20px;
            width: fit-content;
        }

        .btn {
            color: #fff;
            background-color: #0066CC;
            padding: 10px 30px;
            border: solid #0066cc 2px;
            box-shadow: rgba(0, 0, 0, 0.15) 0px 2px 8px;
            border-radius: 50px;
            transition: 500ms;
            transform: translateY(0);
            display: flex;
            flex-direction: row;
            align-items: center;
            cursor: pointer;
            margin: auto;
        }

            .btn:hover {
                transition: 500ms;
                padding: 10px 40px;
                transform: translateY(-0px);
                background-color: #5cad4b;
                border: solid #0066cc 2px;
            }
    </style>

    <script>
        var params = new URLSearchParams(window.location.search);
        var url = params.get("url");
        var reason = params.get("reason");
        var user = params.get("user");
        var recipient = "[email protected]";

        function requireUnlock() {
            let req = prompt("Motivo della richiesta:", "");
            if (req == null || req.trim().length < 5) {
                alert("Necessario inserire una motivazione valida per la richiesta");
            }
            else {
                sendUnlock(req);
            }
        }

        function sendUnlock(unlockReason) {
            var subject = "Richiesta sblocco sito web";
            var row1 = "Utente:\t" + user;
            var row2 = "Sito:\t" + url;
            var row3 = "Blocco:\t" + reason;
            var row4 = "Motivo:\t" + unlockReason;
            var message = row1 + "\n" + row2 + "\n" + row3 + "\n" + row4;
            var mailData = encodeURIComponent(message);
            window.location.href = "mailto:" + recipient + "?subject=" + subject + "&body=" + mailData;
        }

        document.addEventListener("DOMContentLoaded", function () {
            document.getElementById("logo").src = imageSrc;
            document.getElementById("reason").appendChild(document.createTextNode(reason));
            document.getElementById("user").appendChild(document.createTextNode(user));
            document.getElementById("host").appendChild(document.createTextNode(url));
        });

    </script>
</head>
<body>
    <div class="outer">
        <div class="inner">
            <img id="logo" width="20%;" />
            <h1>
                <b>ITsBalto Company</b>
            </h1>
            <h2>
                Accesso Negato
            </h2>
            <p style="font-weight: bold;">
                Comunica le informazioni seguenti al supporto o clicca il pulsante per richiedere l'accesso via mail:
            </p>
            <div style="text-align: left; border: 2px solid #d61e1e; padding: 10px;">
                <p id="host"><b>Sito: </b></p>
                <p id="reason"><b>Motivo: </b></p>
                <p id="user"><b>Utente: </b></p>
                <p>
                    <button onclick="requireUnlock()" class="btn" id="btn">RICHIEDI SBLOCCO</button>
                </p>
            </div>
        </div>
    </div>

    <script>
        const imageSrc = "";
    </script>
</body>
</html>