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

218 lines
7.8 KiB
Python
Raw Normal View History

#!/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 sys
2017-05-08 17:13:14 +02:00
from glob import glob
import hashlib
2017-05-07 17:30:53 +02:00
import os
import re
import yaml
from PIL import Image
ignored_source_files = [
"src/flags/un.svg",
"src/flags/gb-wls.svg",
"src/flags/gb-sct.svg",
"src/flags/gb-eng.svg",
"src/flags/gb-nir.svg"
]
2017-05-08 20:48:32 +02:00
placeholder_icon_filenames = {
"brand": "unk.png",
"browsers": "UNK.png",
"devices": "unknown.png",
"os": "UNK.png",
"searchEngines": "xx.png",
"socials": "xx.png"
}
2017-05-18 19:01:48 +02:00
ignore_that_icon_isnt_square = [
"dist/searchEngines/www.x-recherche.com.png",
2017-10-01 19:57:53 +02:00
"dist/plugins/gears.png",
"dist/brand/Landvo.png"
2017-05-18 19:01:48 +02:00
]
build_script_regex = re.compile(r"rm [-rf]+ plugins/Morpheus/icons/(.*)")
2017-05-10 17:28:14 +02:00
min_image_size = 48
2017-05-08 20:48:32 +02:00
placeholder_icon_hash = "398a623a3b0b10eba6d1884b0ff1713ee12aeafaa8efaf67b60a4624f4dce48c"
searchEnginesFile = "vendor/piwik/searchengine-and-social-list/SearchEngines.yml"
socialsEnginesFile = "vendor/piwik/searchengine-and-social-list/Socials.yml"
def load_yaml(file):
with open(file, 'r') as stream:
return yaml.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():
global error
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
2017-05-08 17:13:14 +02:00
for file in glob("src/**/*.{}".format(filetype)):
2017-05-07 17:30:53 +02:00
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)
2017-05-07 17:30:53 +02:00
if not os.path.isfile(distfile) and file not in ignored_source_files:
print("{file} is missing (From {source})".format(file=distfile, source=file))
error = True
return True
2017-05-07 17:30:53 +02:00
def test_if_source_for_images():
global error
for icontype in ["brand", "browsers", "os", "plugins", "SEO"]:
for filetype in ["svg", "png", "gif", "jpg", "ico"]:
2017-05-08 17:13:14 +02:00
for source_file in glob("src/{type}/*.{filetype}".format(type=icontype, filetype=filetype)):
2017-05-07 17:30:53 +02:00
if not os.path.islink(source_file):
2017-05-08 18:19:55 +02:00
if not os.path.isfile(source_file + ".source") and "UNK" not in source_file:
2017-05-07 17:30:53 +02:00
print("Source is missing for {file}".format(file=source_file))
error = True
2017-05-08 17:13:14 +02:00
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(
2017-05-08 18:19:55 +02:00
"Symlink doesn't link to file (from {link} to {target}".format(link=file, target=os.readlink(file))
2017-05-08 17:13:14 +02:00
)
error = True
2017-05-08 20:48:32 +02:00
def test_if_placeholder_icon_exist():
global error
for folder, filename in placeholder_icon_filenames.items():
file = "src/{folder}/{filename}".format(folder=folder, filename=filename)
2017-05-10 17:28:14 +02:00
if not (os.path.isfile(file) and hashlib.sha256(open(file, "rb").read()).hexdigest() == placeholder_icon_hash):
2017-05-08 20:48:32 +02:00
print("The placeholder icon {path} is missing or invalid".format(path=file))
error = True
2017-05-10 17:28:14 +02:00
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(
"{file} is smaller ({width}x{height}) that the target size ({target}x{target})".format(
2017-05-10 17:28:14 +02:00
file=source_file,
width=im.size[0],
height=im.size[1],
target=min_image_size
)
)
if filetype in ["jpg", "gif", "ico"]:
print("{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.")
2017-05-10 17:28:14 +02:00
def test_if_dist_icons_are_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]:
print("{file} isn't square ({width}x{height})".format(file=file, width=im.size[0], height=im.size[1]))
2017-05-18 19:01:48 +02:00
if file not in ignore_that_icon_isnt_square:
error = True
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")) \
and file != "./README.md":
print("{file} should be deleted by the build script".format(file=file))
2017-05-23 19:06:56 +02:00
error = True
def test_if_icons_are_indicated_to_be_missing():
for file in glob("src/**/*.missing"):
print("{icon} is missing".format(icon=file[:-8]))
def test_if_icons_are_indicated_to_be_improvable():
for file in glob("src/**/*.todo"):
print("{icon} could be improved".format(icon=file[:-5]))
def look_for_search_and_social_icon(source, mode, outputdir):
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 and "/" not in url), False)
if url and not image_exists(outputdir + url):
print(url)
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/")
2017-05-10 17:28:14 +02:00
if __name__ == "__main__":
error = False
2017-05-10 17:28:14 +02:00
if "TRAVIS_PULL_REQUEST" not in os.environ or not os.environ["TRAVIS_PULL_REQUEST"]:
test_if_all_icons_are_converted()
2017-05-07 17:30:53 +02:00
test_if_source_for_images()
2017-05-08 17:13:14 +02:00
test_if_all_symlinks_are_valid()
2017-05-08 20:48:32 +02:00
test_if_placeholder_icon_exist()
test_if_dist_icons_are_square()
2017-05-10 17:28:14 +02:00
if "TRAVIS" in os.environ and os.environ["TRAVIS"]: # collapse on travis
print("travis_fold:start:small_icons")
2017-05-11 08:31:38 +02:00
print("improvable icons: (click to expand)")
test_if_icons_are_indicated_to_be_missing()
test_if_icons_are_indicated_to_be_improvable()
2017-05-10 17:28:14 +02:00
test_if_icons_are_large_enough()
test_if_all_search_and_social_sites_have_an_icon()
2017-05-10 17:28:14 +02:00
print("travis_fold:end:small_icons")
test_if_build_script_is_deleting_all_unneeded_files()
2017-05-10 17:28:14 +02:00
else:
print()
test_if_all_search_and_social_sites_have_an_icon()
sys.exit(error)