1
0
Fork 0
mirror of https://github.com/matomo-org/matomo-icons.git synced 2024-09-19 17:03:45 +02:00
matomo-icons/tests.py
2019-08-13 16:13:48 +02:00

251 lines
9.9 KiB
Python
Executable file

#!/usr/bin/env python3
# Copyright (C) 2017 Lukas Winkler
#
# This program 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 3 of the License, or (at your option) any later
# version.
#
# This program 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 program. If not, see <http://www.gnu.org/licenses/>.
import hashlib
import json
import os
import re
import sys
from glob import glob
from subprocess import Popen, PIPE
from urllib.parse import urlparse
import yaml
from PIL import Image
build_script_regex = re.compile(r"rm [-rf]+ plugins/Morpheus/icons/(.*)")
min_image_size = 48
placeholder_icon_hash = "398a623a3b0b10eba6d1884b0ff1713ee12aeafaa8efaf67b60a4624f4dce48c"
searchEnginesFile = "vendor/matomo/searchengine-and-social-list/SearchEngines.yml"
socialsEnginesFile = "vendor/matomo/searchengine-and-social-list/Socials.yml"
def print_warning(string):
print("\033[33;1m⚠\033[0m " + string)
def print_error(string):
print("\033[31;1m⚠ " + string + "\033[0m")
def load_yaml(file):
with open(file, 'r') as stream:
return yaml.safe_load(stream)
def image_exists(pathslug):
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
if os.path.isfile(pathslug + "." + filetype):
return True
return False
def test_if_all_icons_are_converted(ignored_source_files):
global error
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
for file in glob("src/**/*.{}".format(filetype)):
abs_dirname, filename = os.path.split(file)
code = os.path.splitext(filename)[0]
distfolder = "dist/" + abs_dirname[4:]
distfile = "{folder}/{code}.png".format(folder=distfolder, code=code)
if not os.path.isfile(distfile) and file not in ignored_source_files:
print_error("{file} is missing (From {source})".format(file=distfile, source=file))
error = True
return True
def test_if_source_for_images():
global error
for icontype in ["brand", "browsers", "os", "plugins", "SEO"]:
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
for source_file in glob("src/{type}/*.{filetype}".format(type=icontype, filetype=filetype)):
if not os.path.islink(source_file):
if not os.path.isfile(source_file + ".source") and "UNK" not in source_file:
print_error("Source is missing for {file}".format(file=source_file))
error = True
def test_if_all_symlinks_are_valid():
global error
for file in glob("src/**/*"):
if os.path.islink(file) and not os.path.exists(file):
print_error(
"Symlink doesn't link to file (from {link} to {target}".format(link=file, target=os.readlink(file))
)
error = True
def test_if_placeholder_icon_exist(placeholder_icon_filenames):
global error
for folder, filename in placeholder_icon_filenames.items():
file = "src/{folder}/{filename}".format(folder=folder, filename=filename)
if not (os.path.isfile(file) and hashlib.sha256(open(file, "rb").read()).hexdigest() == placeholder_icon_hash):
print_error("The placeholder icon {path} is missing or invalid".format(path=file))
error = True
def test_if_icons_are_large_enough():
# ignore searchEngines and socials
for filetype in ["png", "gif", "jpg", "ico"]:
for source_file in glob("src/*/*.{filetype}".format(filetype=filetype)):
im = Image.open(source_file)
if im.size[0] < min_image_size or im.size[1] < min_image_size:
print_warning(
"{file} is smaller ({width}x{height}) that the target size ({target}x{target})".format(
file=source_file,
width=im.size[0],
height=im.size[1],
target=min_image_size
)
)
if filetype in ["jpg", "gif", "ico"]:
print_warning("{file} is saved in a lossy image format ({filetype}). ".format(
file=source_file,
filetype=filetype
) + "Maybe try to find an PNG or SVG from another source.")
def test_if_dist_icons_are_square(ignore_that_icon_isnt_square):
global error
for file in glob("dist/**/*.png"):
if "flags" not in file:
im = Image.open(file)
if im.size[0] != im.size[1]:
string = "{file} isn't square ({width}x{height})".format(file=file, width=im.size[0], height=im.size[1])
if file not in ignore_that_icon_isnt_square:
error = True
print_error(string)
else:
print_warning(string)
def test_if_build_script_is_deleting_all_unneeded_files():
with open("tmp/piwik-package/scripts/build-package.sh") as f:
build_script = f.read()
global error
deleted_files = []
all_files = []
for root, dirs, files in os.walk(".", topdown=False):
for name in files:
all_files.append(os.path.join(root, name))
for name in dirs:
all_files.append(os.path.join(root, name))
for pattern in build_script_regex.findall(build_script):
deleted_files.extend(glob(pattern))
for file in all_files:
if not any(s in file for s in deleted_files) and not (
file.startswith("./dist") or file.startswith("./tmp") or file.startswith("./vendor")
or file.startswith("./node_modules")
) and file != "./README.md":
print_error("{file} should be deleted by the build script".format(file=file))
error = True
def test_if_icons_are_indicated_to_be_improvable():
for file in glob("src/**/*.todo"):
print_warning("{icon} could be improved".format(icon=file[:-5]))
def look_for_search_and_social_icon(source, mode, outputdir):
global error
correct_files = []
for i, element in source.items():
if mode == "searchengines":
search_engine = element[0]
urls = search_engine["urls"]
else:
urls = element
url = next((url for url in urls if "{}" not in url), False)
url = urlparse("https://" + url).netloc
if url and not image_exists(outputdir + url):
print_error("icon for {icon} is missing".format(icon=url))
error = True
correct_files.append(url)
# print(correct_files)
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
for file in glob(outputdir + "*.{ext}".format(ext=filetype)):
domain = os.path.splitext(os.path.basename(file))[0]
if domain not in correct_files and domain != "xx":
print_error("{file} is not necessary".format(file=file))
error = True
def test_if_all_search_and_social_sites_have_an_icon():
look_for_search_and_social_icon(load_yaml(searchEnginesFile), "searchengines", "src/searchEngines/")
look_for_search_and_social_icon(load_yaml(socialsEnginesFile), "socials", "src/socials/")
def test_if_there_are_icons_for_all_device_detector_categories(less_important_device_detector_icons):
global error
process = Popen(["php", "devicedetector.php"], stdout=PIPE)
(output, err) = process.communicate()
process.wait()
categories = json.loads(output)
for icontype, category in categories.items():
for code in category:
if icontype == "brand":
slug = category[code].replace(" ", "_")
else:
slug = code
found = False
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
if os.path.isfile("src/{type}/{slug}.{ext}".format(type=icontype, slug=slug, ext=filetype)):
found = True
if not found:
warning = "icon for {icon} missing (should be at src/{type}/{slug}.{{png|svg}})".format(
type=icontype, icon=category[code], slug=slug
)
print(slug)
if slug in less_important_device_detector_icons[icontype]:
print_warning(warning)
else:
print_error(warning)
error = True
if __name__ == "__main__":
error = False
ignore = load_yaml("tests-ignore.yml")
if "TRAVIS_PULL_REQUEST" not in os.environ or not os.environ["TRAVIS_PULL_REQUEST"]:
test_if_all_icons_are_converted(ignore["ignored_source_files"])
test_if_source_for_images()
test_if_all_symlinks_are_valid()
test_if_placeholder_icon_exist(ignore["placeholder_icon_filenames"])
test_if_dist_icons_are_square(ignore["ignore_that_icon_isnt_square"])
if "TRAVIS" in os.environ and os.environ["TRAVIS"]: # collapse on travis
print("travis_fold:start:improvable_icons")
print("improvable icons: (click to expand)")
test_if_there_are_icons_for_all_device_detector_categories(ignore["less_important_device_detector_icons"])
test_if_icons_are_indicated_to_be_improvable()
test_if_icons_are_large_enough()
print("travis_fold:end:improvable_icons")
test_if_all_search_and_social_sites_have_an_icon()
test_if_build_script_is_deleting_all_unneeded_files()
else:
test_if_there_are_icons_for_all_device_detector_categories(ignore["less_important_device_detector_icons"])
test_if_icons_are_indicated_to_be_improvable()
test_if_icons_are_large_enough()
test_if_all_search_and_social_sites_have_an_icon()
test_if_build_script_is_deleting_all_unneeded_files()
sys.exit(error)