web-dev-qa-db-fra.com

Essayer de valider l'URL en utilisant JavaScript

Je veux valider une URL et afficher un message. Ci-dessous mon code:

$("#pageUrl").keydown(function(){
        $(".status").show();
        var url = $("#pageUrl").val();

        if(isValidURL(url)){

        $.ajax({
            type: "POST",
            url: "demo.php",
            data: "pageUrl="+ url,
            success: function(msg){
                if(msg == 1 ){
                    $(".status").html('<img src="images/success.gif"/><span><strong>SiteID:</strong>12345678901234456</span>');
                }else{
                    $(".status").html('<img src="images/failure.gif"/>');
                }
            }
            });

            }else{

                    $(".status").html('<img src="images/failure.gif"/>');
            }

    });


function isValidURL(url){
    var RegExp = /(ftp|http|https):\/\/(\w+:{0,1}\w*@)?(\S+)(:[0-9]+)?(\/|\/([\w#!:.?+=&%@!\-\/]))?/;

    if(RegExp.test(url)){
        return true;
    }else{
        return false;
    }
} 

Mon problème est que maintenant, il affichera un message d'erreur même lorsque vous entrez une URL correcte jusqu'à ce qu'il corresponde à une expression régulière. Il renvoie true même si l'URL ressemble à "http://wwww".

J'apprécie vos suggestions.

83
vinay

Quelqu'un a mentionné le plugin Jquery Validation, semble excessif si vous voulez juste valider l'URL, voici la ligne de regex du plugin:

return this.optional(element) || /^(https?|ftp):\/\/(((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(\#((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/i.test(value);

Voici d'où ils l'ont tiré: http://projects.scottsplayground.com/iri/

Signalé par @nhahtdh Ceci a été mis à jour pour:

        // Copyright (c) 2010-2013 Diego Perini, MIT licensed
        // https://Gist.github.com/dperini/729294
        // see also https://mathiasbynens.be/demo/url-regex
        // modified to allow protocol-relative URLs
        return this.optional( element ) || /^(?:(?:(?:https?|ftp):)?\/\/)(?:\S+(?::\S*)?@)?(?:(?!(?:10|127)(?:\.\d{1,3}){3})(?!(?:169\.254|192\.168)(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)(?:\.(?:[a-z\u00a1-\uffff0-9]-*)*[a-z\u00a1-\uffff0-9]+)*(?:\.(?:[a-z\u00a1-\uffff]{2,})).?)(?::\d{2,5})?(?:[/?#]\S*)?$/i.test( value );

source: https://github.com/jzaefferer/jquery-validation/blob/c1db10a34c0847c28a5bd30e3ee1117e137ca834/src/core.js#L1349

67
artfulhacker

Il n'est pas pratique d'analyser les URL à l'aide de regex. Une mise en œuvre complète des règles RFC1738 aurait pour résultat une regex extrêmement longue (en supposant que ce soit même possible). Certes, votre expression actuelle échoue avec de nombreuses URL valides et transmet celles qui ne sont pas valides.

Au lieu:

une. utilisez un analyseur d'URL approprié qui respecte les règles réelles. (Je n'en connais pas pour JavaScript; ce serait probablement excessif. Vous pourriez le faire du côté serveur). Ou,

b. Il suffit de couper les espaces de début ou de fin, puis de vérifier que l’un de vos modèles préférés est au premier plan (généralement, «http: //» ou «https: //»), et de le laisser tel quel. Ou,

c. essayez d'utiliser l'URL et voyez ce qui se trouve à la fin, par exemple en lui envoyant une requête HTTP HEAD depuis le côté serveur. Si vous obtenez une erreur 404 ou de connexion, c'est probablement faux.

il retourne vrai même si l'url ressemble à " http: // wwww ".

Eh bien, il s’agit bien d’une URL parfaitement valide.

Si vous voulez vérifier si un nom d’hôte tel que «wwww» existe réellement, vous n’avez pas d’autre choix que de le rechercher dans le DNS. Encore une fois, ce serait un code côté serveur.

47
bobince
function validateURL(textval) {
    var urlregex = /^(https?|ftp):\/\/([a-zA-Z0-9.-]+(:[a-zA-Z0-9.&%$-]+)*@)*((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}|([a-zA-Z0-9-]+\.)*[a-zA-Z0-9-]+\.(com|edu|gov|int|mil|net|org|biz|arpa|info|name|pro|aero|coop|museum|[a-zA-Z]{2}))(:[0-9]+)*(\/($|[a-zA-Z0-9.,?'\\+&%$#=~_-]+))*$/;
    return urlregex.test(textval);
}

Cela peut renvoyer true pour des URL telles que:

http://stackoverflow.com/questions/1303872/url-validation-using-javascript

ou:

http://regexlib.com/DisplayPatterns.aspx?cattabindex=1&categoryId=2
33
Mohesh Mohan

J'ai aussi écrit une fonction de validation d'URL basée sur rfc1738 et rfc3986 pour vérifier les URL http et https. J'essaie de maintenir cette structure modulaire pour qu'elle puisse être mieux maintenue et adaptée à ses propres besoins.

La RegExp sur une ligne est affichée à la fin de ce post.

RegExp accepte les URL HTTP et HTTPS avec un domaine international ou un numéro IPv4. IPv6 n'est pas encore supporté. 

window.isValidURL = (function() {// wrapped in self calling function to prevent global pollution

     //URL pattern based on rfc1738 and rfc3986
    var rg_pctEncoded = "%[0-9a-fA-F]{2}";
    var rg_protocol = "(http|https):\\/\\/";

    var rg_userinfo = "([a-zA-Z0-9$\\-_.+!*'(),;:&=]|" + rg_pctEncoded + ")+" + "@";

    var rg_decOctet = "(25[0-5]|2[0-4][0-9]|[0-1][0-9][0-9]|[1-9][0-9]|[0-9])"; // 0-255
    var rg_ipv4address = "(" + rg_decOctet + "(\\." + rg_decOctet + "){3}" + ")";
    var rg_hostname = "([a-zA-Z0-9\\-\\u00C0-\\u017F]+\\.)+([a-zA-Z]{2,})";
    var rg_port = "[0-9]+";

    var rg_hostport = "(" + rg_ipv4address + "|localhost|" + rg_hostname + ")(:" + rg_port + ")?";

    // chars sets
    // safe           = "$" | "-" | "_" | "." | "+"
    // extra          = "!" | "*" | "'" | "(" | ")" | ","
    // hsegment       = *[ alpha | digit | safe | extra | ";" | ":" | "@" | "&" | "=" | escape ]
    var rg_pchar = "a-zA-Z0-9$\\-_.+!*'(),;:@&=";
    var rg_segment = "([" + rg_pchar + "]|" + rg_pctEncoded + ")*";

    var rg_path = rg_segment + "(\\/" + rg_segment + ")*";
    var rg_query = "\\?" + "([" + rg_pchar + "/?]|" + rg_pctEncoded + ")*";
    var rg_fragment = "\\#" + "([" + rg_pchar + "/?]|" + rg_pctEncoded + ")*";

    var rgHttpUrl = new RegExp( 
        "^"
        + rg_protocol
        + "(" + rg_userinfo + ")?"
        + rg_hostport
        + "(\\/"
        + "(" + rg_path + ")?"
        + "(" + rg_query + ")?"
        + "(" + rg_fragment + ")?"
        + ")?"
        + "$"
    );

    // export public function
    return function (url) {
        if (rgHttpUrl.test(url)) {
            return true;
        } else {
            return false;
        }
    };
})();

RegExp en une ligne:

var rg = /^(http|https):\/\/(([a-zA-Z0-9$\-_.+!*'(),;:&=]|%[0-9a-fA-F]{2})+@)?(((25[0-5]|2[0-4][0-9]|[0-1][0-9][0-9]|[1-9][0-9]|[0-9])(\.(25[0-5]|2[0-4][0-9]|[0-1][0-9][0-9]|[1-9][0-9]|[0-9])){3})|localhost|([a-zA-Z0-9\-\u00C0-\u017F]+\.)+([a-zA-Z]{2,}))(:[0-9]+)?(\/(([a-zA-Z0-9$\-_.+!*'(),;:@&=]|%[0-9a-fA-F]{2})*(\/([a-zA-Z0-9$\-_.+!*'(),;:@&=]|%[0-9a-fA-F]{2})*)*)?(\?([a-zA-Z0-9$\-_.+!*'(),;:@&=\/?]|%[0-9a-fA-F]{2})*)?(\#([a-zA-Z0-9$\-_.+!*'(),;:@&=\/?]|%[0-9a-fA-F]{2})*)?)?$/;
15
TLindig

Dans une situation similaire, je suis parti avec ceci:

someUtils.validateURL = function(url) {
    var parser = document.createElement('a');
    try {
        parser.href = url;
        return !!parser.hostname;
    } catch (e) {
        return false;
    }
};

pourquoi inventer la roue si les navigateurs peuvent le faire pour vous? Mais, bien sûr, cela ne fonctionnera que dans le navigateur. 

il existe différentes parties de l'URL analysée exactement comment le navigateur l'interprète:

parser.protocol; // => "http:"
parser.hostname; // => "example.com"
parser.port;     // => "8080"
parser.pathname; // => "/path/"
parser.search;   // => "?search=test"
parser.hash;     // => "#hash"
parser.Host;     // => "example.com:3000"

En les utilisant, vous pouvez améliorer votre fonction de validation en fonction des besoins. Le seul inconvénient est qu'il acceptera les URL relatives et utilisera l'hôte et le port du serveur de pages actuel. Mais vous pouvez l'utiliser à votre avantage en réassemblant l'URL à partir de pièces et en la transmettant toujours intégralement à votre service AJAX.

Ce que validateURL n'acceptera pas est une URL non valide, par exemple. http:\:8883 retournera false, mais :1234 est valide et est interprété comme http://pagehost.example.com/:1234 c'est-à-dire comme un chemin relatif. 

METTRE À JOUR

Cette approche ne fonctionne plus avec Chrome ni avec les autres navigateurs WebKit. Même lorsque l'URL n'est pas valide, le nom d'hôte est rempli avec une valeur, par exemple. pris de base. Il est toujours utile d’analyser certaines parties de l’URL, mais ne permet pas de en valider une.

Une meilleure approche possible sans analyseur propriétaire consiste à utiliser var parsedURL = new URL(url) et à capturer des exceptions. Voir par exemple URL API . Pris en charge par tous les principaux navigateurs et NodeJS, bien que toujours marqué comme expérimental.

12
Alex Pakka

meilleure expression rationnelle que j'ai trouvée de http://angularjs.org/

var urlregex = /^(ftp|http|https):\/\/(\w+:{0,1}\w*@)?(\S+)(:[0-9]+)?(\/|\/([\w#!:.?+=&%@!\-\/]))?$/;

8
nguyên

Je sais que c’est une question assez ancienne, mais comme elle n’a pas de réponse acceptée, je vous suggère d’utiliser le cadre URI.js: https://github.com/medialize/URI.js

Vous pouvez l'utiliser pour vérifier l'URI mal formé à l'aide d'un bloc try/catch:

function isValidURL(url)
{
    try {
        (new URI(url));
        return true;
    }
    catch (e) {
        // Malformed URI
        return false;
    }
}

Bien sûr, il considérera quelque chose comme "% @" en tant qu'URI relatif bien formé ... Je vous suggère donc de lire l'URI.js API pour effectuer plusieurs vérifications, par exemple si vous voulez vous assurer que l'utilisateur a entré une URL absolue bien formée, vous pouvez le faire comme ceci:

function isValidURL(url)
{
    try {
        var uri = new URI(url);
        // URI has a scheme and a Host
        return (!!uri.scheme() && !!uri.Host());
    }
    catch (e) {
        // Malformed URI
        return false;
    }
}
6
Romain

C'est ce qui a fonctionné pour moi:

function validateURL(value) {
    return /^(https?|ftp):\/\/(((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(\#((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/i.test(value);
    }

à partir de là, il suffit d'appeler la fonction pour obtenir un retour vrai ou faux:

validateURL(urltovalidate);
3
Crisman

Importer dans un paquet npm comme

https://www.npmjs.com/package/valid-url

et l'utiliser pour valider votre URL.

2
ooolala
var RegExp = (/^HTTP|HTTP|http(s)?:\/\/(www\.)?[A-Za-z0-9]+([\-\.]{1}[A-Za-z0-9]+)*\.[A-Za-z]{2,40}(:[0-9]{1,40})?(\/.*)?$/);
1
SagarPPanchal

Vous pouvez utiliser le URL API qui est récemment devenu standard. Le support du navigateur est au mieux incomplet, voir le lien. new URL(str) a la garantie de générer TypeError pour les URL non valides.

Comme indiqué ci-dessus, http://wwww est une URL valide.

1
amoe

Si vous recherchez un regex plus fiable, consultez RegexLib. Voici la page qui vous intéresserait probablement:

http://regexlib.com/Search.aspx?k=url

En ce qui concerne les messages d'erreur indiquant que la personne est encore en train de taper, changez l'événement de keydown à blur et il ne vérifiera que lorsque la personne passera à l'élément suivant.

1
nickf

Ma solution:

function isValidUrl(t)
{
    return t.match(/^(http|https|ftp):\/\/(([A-Z0-9][A-Z0-9_-]*)(\.[A-Z0-9][A-Z0-9_-]*)+)(:(\d+))?\/?/i)
}
1
Dung Vu

J'ai vérifié beaucoup de validateurs d'URL dans Google et personne ne travaille pour moi. Par exemple, j'aimerais que les liens comme "aa.com" soient valides. J'aime vérifier stupide pour le signe de point dans la chaîne.

function isValidUri(str) {
  var dotIndex = str.indexOf('.');
  return (dotIndex > 0 && dotIndex < str.length - 2);
}

Il ne doit pas rester au début et à la fin de la chaîne (pour l'instant, nous n'avons pas de noms de domaine de premier niveau avec un seul caractère). 

0
alex naumov

Démo: http://jsbin.com/uzimeb/1/edit

function checkURL(value) {
    var urlregex = new RegExp("^(http|https|ftp)\://([a-zA-Z0-9\.\-]+(\:[a-zA-Z0-9\.&amp;%\$\-]+)*@)*((25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9])\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9]|0)\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[1-9]|0)\.(25[0-5]|2[0-4][0-9]|[0-1]{1}[0-9]{2}|[1-9]{1}[0-9]{1}|[0-9])|([a-zA-Z0-9\-]+\.)*[a-zA-Z0-9\-]+\.(com|edu|gov|int|mil|net|org|biz|arpa|info|name|pro|aero|coop|museum|[a-zA-Z]{2}))(\:[0-9]+)*(/($|[a-zA-Z0-9\.\,\?\'\\\+&amp;%\$#\=~_\-]+))*$");
    if (urlregex.test(value)) {
        return (true);
    }
    return (false);
}
0
Yasser

J'ai trouvé une excellente ressource pour comparer différentes solutions: https://mathiasbynens.be/demo/url-regex

Selon cette page, seule la solution de diegoperini passe tous les tests. Voici cette regex:

_^(?:(?:https?|ftp)://)(?:\S+(?::\S*)?@)?(?:(?!10(?:\.\d{1,3}){3})(?!127(?:\.\d{1,3}){3})(?!169\.254(?:\.\d{1,3}){2})(?!192\.168(?:\.\d{1,3}){2})(?!172\.(?:1[6-9]|2\d|3[0-1])(?:\.\d{1,3}){2})(?:[1-9]\d?|1\d\d|2[01]\d|22[0-3])(?:\.(?:1?\d{1,2}|2[0-4]\d|25[0-5])){2}(?:\.(?:[1-9]\d?|1\d\d|2[0-4]\d|25[0-4]))|(?:(?:[a-z\x{00a1}-\x{ffff}0-9]+-?)*[a-z\x{00a1}-\x{ffff}0-9]+)(?:\.(?:[a-z\x{00a1}-\x{ffff}0-9]+-?)*[a-z\x{00a1}-\x{ffff}0-9]+)*(?:\.(?:[a-z\x{00a1}-\x{ffff}]{2,})))(?::\d{2,5})?(?:/[^\s]*)?$_iuS
0
Zoran P.