2013-08-28 12:17:40 +02:00
|
|
|
#! /usr/bin/env perl
|
|
|
|
# -*- mode: perl; -*-
|
|
|
|
#
|
|
|
|
# file search_url.pl
|
|
|
|
# script to search for url's in lyxfiles
|
|
|
|
# and testing their validity.
|
|
|
|
#
|
|
|
|
# Syntax: search_url.pl [(filesToScan|(ignored|reverted|extra|selected)URLS)={path_to_control]*
|
|
|
|
# Param value is a path to a file containing list of xxx:
|
|
|
|
# filesToScan={xxx = lyx-file-names to be scanned for}
|
|
|
|
# ignoredURLS={xxx = urls that are discarded from test}
|
|
|
|
# revertedURLS={xxx = urls that should fail, to test the test with invalid urls}
|
|
|
|
# extraURLS={xxx = urls which should be also checked}
|
|
|
|
#
|
|
|
|
# This file is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public
|
|
|
|
# License as published by the Free Software Foundation; either
|
|
|
|
# version 2 of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This software is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
# General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public
|
|
|
|
# License along with this software; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
#
|
|
|
|
# Copyright (c) 2013 Kornel Benko <kornel@lyx.org>
|
|
|
|
# (c) 2013 Scott Kostyshak <skotysh@lyx.org>
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
|
|
|
|
BEGIN {
|
|
|
|
use File::Spec;
|
|
|
|
my $p = File::Spec->rel2abs(__FILE__);
|
|
|
|
$p =~ s/[\/\\]?[^\/\\]+$//;
|
|
|
|
unshift(@INC, "$p");
|
|
|
|
}
|
|
|
|
|
|
|
|
use CheckURL;
|
2014-01-08 13:36:30 +01:00
|
|
|
use Try::Tiny;
|
|
|
|
use locale;
|
|
|
|
use POSIX qw(locale_h);
|
2013-08-28 12:17:40 +02:00
|
|
|
|
2014-01-08 13:36:30 +01:00
|
|
|
setlocale(LC_CTYPE, "");
|
|
|
|
setlocale(LC_MESSAGES, "en_US.UTF-8");
|
2013-08-28 12:17:40 +02:00
|
|
|
|
2014-01-24 11:00:07 +01:00
|
|
|
# Prototypes
|
|
|
|
sub printNotUsedURLS($\%);
|
|
|
|
sub readUrls($\%);
|
|
|
|
sub parse_file($ );
|
|
|
|
sub handle_url($$$ );
|
|
|
|
##########
|
|
|
|
|
2013-08-28 12:17:40 +02:00
|
|
|
my %URLS = ();
|
|
|
|
my %ignoredURLS = ();
|
|
|
|
my %revertedURLS = ();
|
|
|
|
my %extraURLS = ();
|
|
|
|
my %selectedURLS = ();
|
|
|
|
|
|
|
|
my $checkSelectedOnly = 0;
|
|
|
|
for my $arg (@ARGV) {
|
|
|
|
die("Bad argument \"$arg\"") if ($arg !~ /=/);
|
|
|
|
my ($type,$val) = split("=", $arg);
|
|
|
|
if ($type eq "filesToScan") {
|
|
|
|
#The file should be a list of files to search in
|
|
|
|
if (open(FLIST, $val)) {
|
|
|
|
while (my $l = <FLIST>) {
|
|
|
|
chomp($l);
|
2014-01-24 11:00:07 +01:00
|
|
|
parse_file($l);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
close(FLIST);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($type eq "ignoredURLS") {
|
2014-01-24 11:00:07 +01:00
|
|
|
readUrls($val, %ignoredURLS);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
elsif ($type eq "revertedURLS") {
|
2014-01-24 11:00:07 +01:00
|
|
|
readUrls($val, %revertedURLS);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
elsif ($type eq "extraURLS") {
|
2014-01-24 11:00:07 +01:00
|
|
|
readUrls($val, %extraURLS);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
elsif ($type eq "selectedURLS") {
|
|
|
|
$checkSelectedOnly = 1;
|
2014-01-24 11:00:07 +01:00
|
|
|
readUrls($val, %selectedURLS);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
die("Invalid argument \"$arg\"");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my @urls = sort keys %URLS, keys %extraURLS;
|
|
|
|
my $errorcount = 0;
|
|
|
|
|
|
|
|
my $URLScount = 0;
|
|
|
|
|
|
|
|
for my $u (@urls) {
|
2013-08-30 17:28:46 +02:00
|
|
|
if (defined($selectedURLS{$u})) {
|
|
|
|
${selectedURLS}{$u}->{count} += 1;
|
|
|
|
}
|
2013-08-28 16:50:23 +02:00
|
|
|
if (defined($ignoredURLS{$u})) {
|
2013-08-30 17:28:46 +02:00
|
|
|
$ignoredURLS{$u}->{count} += 1;
|
2013-08-28 16:50:23 +02:00
|
|
|
next;
|
|
|
|
}
|
2013-08-30 17:28:46 +02:00
|
|
|
next if ($checkSelectedOnly && ! defined($selectedURLS{$u}));
|
2013-08-28 12:17:40 +02:00
|
|
|
$URLScount++;
|
2014-01-08 13:36:30 +01:00
|
|
|
print "Checking '$u': ";
|
|
|
|
my ($res, $prnt);
|
|
|
|
try {
|
2014-01-24 11:00:07 +01:00
|
|
|
$res = check_url($u);
|
2014-01-08 13:36:30 +01:00
|
|
|
if ($res) {
|
|
|
|
$prnt = "Failed";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$prnt = "OK";
|
|
|
|
}
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
2014-01-08 13:36:30 +01:00
|
|
|
catch {
|
|
|
|
$prnt = "Failed, caught error: $_";
|
|
|
|
$res = 700;
|
|
|
|
};
|
|
|
|
print "$prnt\n";
|
2013-08-28 12:17:40 +02:00
|
|
|
my $printSourceFiles = 0;
|
|
|
|
my $err_txt = "Error url:";
|
|
|
|
|
|
|
|
if ($res || $checkSelectedOnly) {
|
|
|
|
$printSourceFiles = 1;
|
|
|
|
}
|
|
|
|
if ($res && defined($revertedURLS{$u})) {
|
|
|
|
$err_txt = "Failed url:";
|
|
|
|
}
|
|
|
|
$res = ! $res if (defined($revertedURLS{$u}));
|
|
|
|
if ($res || $checkSelectedOnly) {
|
|
|
|
print "$err_txt \"$u\"\n";
|
|
|
|
}
|
|
|
|
if ($printSourceFiles) {
|
|
|
|
if (defined($URLS{$u})) {
|
|
|
|
for my $f(sort keys %{$URLS{$u}}) {
|
2013-08-28 19:51:19 +02:00
|
|
|
my $lines = ":" . join(',', @{$URLS{$u}->{$f}});
|
|
|
|
print " $f$lines\n";
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($res ) {
|
|
|
|
$errorcount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-30 17:28:46 +02:00
|
|
|
if (%URLS) {
|
2014-01-24 11:00:07 +01:00
|
|
|
printNotUsedURLS("Ignored", %ignoredURLS);
|
|
|
|
printNotUsedURLS("Selected", %selectedURLS);
|
|
|
|
printNotUsedURLS("KnownInvalid", %extraURLS);
|
2013-08-30 17:28:46 +02:00
|
|
|
}
|
2013-08-28 16:50:23 +02:00
|
|
|
|
2013-08-28 12:17:40 +02:00
|
|
|
print "\n$errorcount URL-tests failed out of $URLScount\n\n";
|
|
|
|
exit($errorcount);
|
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
2014-01-24 11:00:07 +01:00
|
|
|
sub printNotUsedURLS($\%)
|
2013-08-28 16:50:23 +02:00
|
|
|
{
|
|
|
|
my ($txt, $rURLS) = @_;
|
|
|
|
my @msg = ();
|
|
|
|
for my $u ( sort keys %{$rURLS}) {
|
2013-08-30 17:28:46 +02:00
|
|
|
if ($rURLS->{$u}->{count} < 2) {
|
|
|
|
my @submsg = ();
|
|
|
|
for my $f (sort keys %{$rURLS->{$u}}) {
|
|
|
|
next if ($f eq "count");
|
|
|
|
push(@submsg, "$f:" . $rURLS->{$u}->{$f});
|
|
|
|
}
|
|
|
|
push(@msg, "\n $u\n " . join("\n ", @submsg) . "\n");
|
2013-08-28 16:50:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (@msg) {
|
|
|
|
print "\n$txt URLs not found in sources: " . join(' ',@msg) . "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-24 11:00:07 +01:00
|
|
|
sub readUrls($\%)
|
2013-08-28 12:17:40 +02:00
|
|
|
{
|
|
|
|
my ($file, $rUrls) = @_;
|
|
|
|
|
|
|
|
die("Could not read file $file") if (! open(ULIST, $file));
|
2013-08-30 17:28:46 +02:00
|
|
|
my $line = 0;
|
2013-08-28 12:17:40 +02:00
|
|
|
while (my $l = <ULIST>) {
|
2013-08-30 17:28:46 +02:00
|
|
|
$line++;
|
2013-08-28 12:17:40 +02:00
|
|
|
$l =~ s/[\r\n]+$//; # remove eol
|
|
|
|
$l =~ s/\s*\#.*$//; # remove comment
|
|
|
|
next if ($l eq "");
|
2013-08-30 17:28:46 +02:00
|
|
|
if (! defined($rUrls->{$l} )) {
|
|
|
|
$rUrls->{$l} = {$file => $line, count => 1};
|
|
|
|
}
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
close(ULIST);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub parse_file($)
|
|
|
|
{
|
|
|
|
my($f) = @_;
|
2013-08-30 17:28:46 +02:00
|
|
|
my $status = "out"; # outside of URL/href
|
2013-08-28 12:17:40 +02:00
|
|
|
|
|
|
|
return if ($f =~ /\/attic\//);
|
|
|
|
if(open(FI, $f)) {
|
2013-08-28 19:51:19 +02:00
|
|
|
my $line = 0;
|
2013-08-28 12:17:40 +02:00
|
|
|
while(my $l = <FI>) {
|
2013-08-28 19:51:19 +02:00
|
|
|
$line++;
|
2013-08-28 12:17:40 +02:00
|
|
|
$l =~ s/[\r\n]+$//; # Simulate chomp
|
2013-08-28 19:51:19 +02:00
|
|
|
if ($status eq "out") {
|
2013-08-28 12:17:40 +02:00
|
|
|
# searching for "\begin_inset Flex URL"
|
|
|
|
if($l =~ /^\s*\\begin_inset\s+Flex\s+URL\s*$/) {
|
2013-08-28 19:51:19 +02:00
|
|
|
$status = "inUrlInset";
|
|
|
|
}
|
|
|
|
elsif ($l =~ /^\s*\\begin_inset\s+CommandInset\s+href\s*$/) {
|
|
|
|
$status = "inHrefInset";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
# Outside of url, check also
|
|
|
|
if ($l =~ /"((ftp|http|https):\/\/[^ ]+)"/) {
|
|
|
|
my $url = $1;
|
2014-01-24 11:00:07 +01:00
|
|
|
handle_url($url, $f, "x$line");
|
2013-08-28 19:51:19 +02:00
|
|
|
}
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if($l =~ /^\s*\\end_(layout|inset)\s*$/) {
|
|
|
|
$status = "out";
|
|
|
|
}
|
2013-08-28 19:51:19 +02:00
|
|
|
elsif ($status eq "inUrlInset") {
|
|
|
|
if ($l =~ /\s*([a-z]+:\/\/.+)\s*$/) {
|
|
|
|
my $url = $1;
|
|
|
|
$status = "out";
|
2014-01-24 11:00:07 +01:00
|
|
|
handle_url($url, $f, "u$line");
|
2013-08-28 19:51:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($status eq "inHrefInset") {
|
|
|
|
if ($l =~ /^target\s+"([a-z]+:\/\/[^ ]+)"$/) {
|
2013-08-28 12:17:40 +02:00
|
|
|
my $url = $1;
|
|
|
|
$status = "out";
|
2014-01-24 11:00:07 +01:00
|
|
|
handle_url($url, $f, "h$line");
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(FI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 19:51:19 +02:00
|
|
|
sub handle_url($$$)
|
2013-08-28 12:17:40 +02:00
|
|
|
{
|
2013-08-28 19:51:19 +02:00
|
|
|
my($url, $f, $line) = @_;
|
2013-08-28 12:17:40 +02:00
|
|
|
|
|
|
|
if(!defined($URLS{$url})) {
|
|
|
|
$URLS{$url} = {};
|
2014-01-24 11:00:07 +01:00
|
|
|
}
|
|
|
|
if(!defined($URLS{$url}->{$f})) {
|
2013-08-28 19:51:19 +02:00
|
|
|
$URLS{$url}->{$f} = [];
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|
2013-08-28 19:51:19 +02:00
|
|
|
push(@{$URLS{$url}->{$f}}, $line);
|
2013-08-28 12:17:40 +02:00
|
|
|
}
|