2016-05-25 15:53:13 +02:00
|
|
|
#!/usr/bin/env python
|
2016-03-10 18:22:27 +01:00
|
|
|
from __future__ import absolute_import
|
2016-07-08 22:25:20 +02:00
|
|
|
from __future__ import print_function
|
2016-08-02 00:14:01 +02:00
|
|
|
from lib.template_parser import validate
|
2017-01-06 15:11:15 +01:00
|
|
|
from lib.html_branches import build_id_dict
|
2017-03-12 22:24:26 +01:00
|
|
|
from lib.pretty_print import validate_indent_html
|
2016-08-29 00:28:59 +02:00
|
|
|
import argparse
|
2013-10-25 23:46:02 +02:00
|
|
|
import sys
|
2017-01-06 15:11:15 +01:00
|
|
|
import logging
|
2016-08-02 00:14:01 +02:00
|
|
|
|
2016-03-10 18:22:27 +01:00
|
|
|
from six.moves import filter
|
2017-02-05 21:24:28 +01:00
|
|
|
|
|
|
|
# check for the venv
|
|
|
|
from lib import sanity_check
|
|
|
|
sanity_check.check_venv(__file__)
|
|
|
|
|
|
|
|
import lister
|
|
|
|
from typing import cast, Callable, Dict, Iterable, List
|
2013-10-25 23:46:02 +02:00
|
|
|
|
2017-01-06 15:11:15 +01:00
|
|
|
EXCLUDED_FILES = [
|
|
|
|
## Test data Files for testing modules in tests
|
|
|
|
"tools/tests/test_template_data",
|
|
|
|
]
|
|
|
|
|
2014-02-27 16:41:47 +01:00
|
|
|
def check_our_files():
|
2016-07-24 15:24:48 +02:00
|
|
|
# type: () -> None
|
2016-08-29 00:28:59 +02:00
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('-m', '--modified',
|
2016-12-11 14:30:45 +01:00
|
|
|
action='store_true', default=False,
|
|
|
|
help='only check modified files')
|
2017-01-06 15:11:15 +01:00
|
|
|
parser.add_argument('--all-dups',
|
|
|
|
action="store_true", default=False,
|
|
|
|
help='Run lint tool to detect duplicate ids on ignored files as well')
|
2016-08-29 00:28:59 +02:00
|
|
|
args = parser.parse_args()
|
2016-07-08 22:40:24 +02:00
|
|
|
|
2016-07-24 15:24:48 +02:00
|
|
|
by_lang = cast(
|
|
|
|
Dict[str, List[str]],
|
|
|
|
lister.list_files(
|
2016-08-29 00:28:59 +02:00
|
|
|
modified_only=args.modified,
|
2016-07-24 15:24:48 +02:00
|
|
|
ftypes=['handlebars', 'html'],
|
2017-01-06 15:11:15 +01:00
|
|
|
group_by_ftype=True, exclude=EXCLUDED_FILES))
|
2013-10-25 20:47:03 +02:00
|
|
|
|
2016-08-29 00:28:59 +02:00
|
|
|
check_handlebar_templates(by_lang['handlebars'], args.modified)
|
2017-01-06 15:11:15 +01:00
|
|
|
check_html_templates(by_lang['html'], args.modified, args.all_dups)
|
2013-11-18 22:57:55 +01:00
|
|
|
|
2017-01-06 15:11:15 +01:00
|
|
|
def check_html_templates(templates, modified_only, all_dups):
|
|
|
|
# type: (Iterable[str], bool, bool) -> None
|
2016-07-09 02:43:32 +02:00
|
|
|
# Our files with .html extensions are usually for Django, but we also
|
2016-07-12 21:28:08 +02:00
|
|
|
# have a few static .html files.
|
2016-07-09 02:43:32 +02:00
|
|
|
# The file base.html has a bit of funny HTML that we can't parse here yet.
|
2016-07-12 21:28:08 +02:00
|
|
|
#
|
|
|
|
# We also have .html files that we vendored from Casper.
|
|
|
|
# The casperjs files use HTML5 (whereas Zulip prefers XHTML), and
|
|
|
|
# there are also cases where Casper deliberately uses invalid HTML,
|
|
|
|
# so we exclude them from our linter.
|
2017-01-06 15:11:15 +01:00
|
|
|
logging.basicConfig(format='%(levelname)s:%(message)s')
|
2016-07-12 21:28:08 +02:00
|
|
|
templates = filter(
|
2017-01-06 15:11:15 +01:00
|
|
|
lambda fn: (('base.html' not in fn) and ('casperjs' not in fn)),
|
2016-07-12 21:28:08 +02:00
|
|
|
templates)
|
2016-07-09 02:43:32 +02:00
|
|
|
templates = sorted(list(templates))
|
2013-10-25 20:47:03 +02:00
|
|
|
|
2017-01-06 15:11:15 +01:00
|
|
|
template_id_dict = build_id_dict(templates)
|
|
|
|
# TODO: Clean up these cases of duplicate ids in the code
|
|
|
|
IGNORE_IDS = [
|
|
|
|
'api-example-tabs',
|
|
|
|
'errors',
|
|
|
|
'email',
|
2017-01-22 07:44:13 +01:00
|
|
|
'messages',
|
2017-01-06 15:11:15 +01:00
|
|
|
'registration',
|
|
|
|
'pw_strength',
|
|
|
|
'id_password',
|
|
|
|
'top_navbar',
|
|
|
|
'id_email',
|
|
|
|
'id_terms',
|
|
|
|
'send_confirm',
|
|
|
|
'register',
|
|
|
|
]
|
|
|
|
bad_ids_dict = {ids: fns for ids, fns in template_id_dict.items()
|
2017-01-24 06:08:52 +01:00
|
|
|
if (ids not in IGNORE_IDS) and len(fns) > 1}
|
2017-01-06 15:11:15 +01:00
|
|
|
|
|
|
|
if all_dups:
|
|
|
|
ignorable_ids_dict = {ids: fns for ids, fns in template_id_dict.items()
|
|
|
|
if ids in IGNORE_IDS and len(fns) > 1}
|
|
|
|
|
|
|
|
for ids, fns in ignorable_ids_dict.items():
|
2017-01-24 05:50:04 +01:00
|
|
|
logging.warning("Duplicate ID(s) detected :Id '" + ids +
|
|
|
|
"' present at following files:")
|
2017-01-06 15:11:15 +01:00
|
|
|
for fn in fns:
|
|
|
|
print(fn)
|
|
|
|
|
|
|
|
for ids, fns in bad_ids_dict.items():
|
2017-01-24 05:50:04 +01:00
|
|
|
logging.error("Duplicate ID(s) detected :Id '" + ids +
|
|
|
|
"' present at following files:")
|
2017-01-06 15:11:15 +01:00
|
|
|
for fn in fns:
|
|
|
|
print(fn)
|
|
|
|
|
|
|
|
if list(bad_ids_dict.keys()):
|
|
|
|
print('Exiting--please clean up all duplicates before running this again.')
|
|
|
|
sys.exit(1)
|
|
|
|
|
2016-07-08 22:40:24 +02:00
|
|
|
if not modified_only:
|
|
|
|
assert len(templates) >= 10 # sanity check that we are actually doing work
|
2014-02-27 16:41:47 +01:00
|
|
|
for fn in templates:
|
2016-07-12 22:32:36 +02:00
|
|
|
# Many of our Django templates have strange indentation. The
|
|
|
|
# indentation errors are often harmless, even stylistically
|
|
|
|
# harmless, but they tend to be in files that might be old
|
|
|
|
# and might eventually require more scrutiny for things like
|
|
|
|
# localization. See github #1236.
|
|
|
|
bad_files = [
|
|
|
|
'static/html/5xx.html',
|
|
|
|
'templates/500.html',
|
|
|
|
'templates/confirmation/confirm.html',
|
|
|
|
'templates/corporate/mit.html',
|
|
|
|
'templates/corporate/privacy.html',
|
|
|
|
'templates/corporate/zephyr-mirror.html',
|
|
|
|
'templates/corporate/zephyr.html',
|
|
|
|
'templates/zerver/accounts_home.html',
|
|
|
|
'templates/zerver/accounts_send_confirm.html',
|
|
|
|
'templates/zerver/api.html',
|
|
|
|
'templates/zerver/api_endpoints.html',
|
|
|
|
'templates/zerver/apps.html',
|
|
|
|
'templates/zerver/create_realm.html',
|
|
|
|
'templates/zerver/emails/followup/day1.html',
|
|
|
|
'templates/zerver/emails/followup/day2.html',
|
|
|
|
'templates/zerver/features.html',
|
|
|
|
'templates/zerver/hello.html',
|
|
|
|
'templates/zerver/home.html',
|
|
|
|
'templates/zerver/integrations.html',
|
|
|
|
'templates/zerver/invite_user.html',
|
|
|
|
'templates/zerver/login.html',
|
|
|
|
'templates/zerver/markdown_help.html',
|
|
|
|
'templates/zerver/register.html',
|
|
|
|
'templates/zerver/search_operators.html',
|
2016-11-26 00:25:05 +01:00
|
|
|
|
|
|
|
'zerver/webhooks/deskdotcom/doc.html',
|
|
|
|
'zerver/webhooks/freshdesk/doc.html',
|
|
|
|
'zerver/webhooks/taiga/doc.html',
|
|
|
|
'zerver/webhooks/zendesk/doc.html',
|
2016-07-12 22:32:36 +02:00
|
|
|
]
|
2016-08-04 01:44:15 +02:00
|
|
|
validate(fn=fn, check_indent=(fn not in bad_files))
|
2014-02-27 16:41:47 +01:00
|
|
|
|
2017-03-12 22:24:26 +01:00
|
|
|
# Ignore these files since these have not been cleaned yet :/
|
|
|
|
IGNORE_FILES = [
|
|
|
|
'puppet/zulip_ops/files/sparkle/mac/sparkle-changelog.html',
|
|
|
|
'puppet/zulip_ops/files/sparkle/sso/mac/sparkle-changelog.html',
|
|
|
|
'puppet/zulip_ops/files/sparkle/sso/win/sparkle-changelog.html',
|
|
|
|
'puppet/zulip_ops/files/sparkle/win/sparkle-changelog.html',
|
|
|
|
'static/html/404.html',
|
|
|
|
'static/html/5xx.html',
|
|
|
|
'templates/500.html',
|
|
|
|
'templates/analytics/activity.html',
|
|
|
|
'templates/analytics/ad_hoc_query.html',
|
|
|
|
'templates/analytics/realm_summary_table.html',
|
|
|
|
'templates/analytics/stats.html',
|
|
|
|
'templates/confirmation/confirm.html',
|
|
|
|
'templates/confirmation/confirm_email_change.html',
|
|
|
|
'templates/confirmation/emailchangestatus_confirmation_email.html',
|
|
|
|
'templates/confirmation/invite_email.html',
|
|
|
|
'templates/confirmation/preregistrationuser_confirmation_email.html',
|
|
|
|
'templates/corporate/privacy.html',
|
|
|
|
'templates/corporate/zephyr-mirror.html',
|
|
|
|
'templates/corporate/zephyr.html',
|
|
|
|
'templates/zerver/api.html',
|
|
|
|
'templates/zerver/apps.html',
|
|
|
|
'templates/zerver/compose.html',
|
|
|
|
'templates/zerver/emails/digest/digest_email.html',
|
|
|
|
'templates/zerver/emails/find_team/find_team_email.html',
|
|
|
|
'templates/zerver/emails/followup/day1.html',
|
|
|
|
'templates/zerver/emails/followup/day2.html',
|
|
|
|
'templates/zerver/emails/invitation/invitation_reminder_email.html',
|
|
|
|
'templates/zerver/hello.html',
|
|
|
|
'templates/zerver/help/main.html',
|
|
|
|
'templates/zerver/home.html',
|
|
|
|
'templates/zerver/index.html',
|
|
|
|
'templates/zerver/integrations.html',
|
|
|
|
'templates/zerver/keyboard_shortcuts.html',
|
|
|
|
'templates/zerver/landing_nav.html',
|
|
|
|
'templates/zerver/left_sidebar.html',
|
|
|
|
'templates/zerver/login.html',
|
|
|
|
'templates/zerver/markdown_help.html',
|
|
|
|
'templates/zerver/navbar.html',
|
|
|
|
'templates/zerver/register.html',
|
|
|
|
'zerver/webhooks/deskdotcom/doc.html',
|
|
|
|
'zerver/webhooks/greenhouse/doc.html',
|
|
|
|
'zerver/webhooks/librato/doc.html',
|
|
|
|
'zerver/webhooks/pivotal/doc.html',
|
|
|
|
'zerver/webhooks/splunk/doc.html',
|
|
|
|
'zerver/webhooks/stripe/doc.html',
|
|
|
|
'zerver/webhooks/trello/doc.html',
|
|
|
|
'zerver/webhooks/wordpress/doc.html',
|
|
|
|
'zerver/webhooks/zapier/doc.html',
|
|
|
|
]
|
|
|
|
# TODO: Clean these files
|
|
|
|
for fn in templates:
|
|
|
|
if fn not in IGNORE_FILES:
|
|
|
|
if not validate_indent_html(fn):
|
|
|
|
sys.exit(1)
|
|
|
|
|
2016-08-02 00:14:01 +02:00
|
|
|
def check_handlebar_templates(templates, modified_only):
|
|
|
|
# type: (Iterable[str], bool) -> None
|
|
|
|
# Check all our handlebars templates.
|
|
|
|
templates = [fn for fn in templates if fn.endswith('.handlebars')]
|
|
|
|
if not modified_only:
|
|
|
|
assert len(templates) >= 10 # sanity check that we are actually doing work
|
|
|
|
for fn in templates:
|
2016-08-04 01:44:15 +02:00
|
|
|
validate(fn=fn, check_indent=True)
|
2016-08-02 00:14:01 +02:00
|
|
|
|
2017-03-12 22:24:26 +01:00
|
|
|
# Ignore these files since these have not been cleaned yet :/
|
|
|
|
IGNORE_FILES = [
|
|
|
|
'static/templates/actions_popover_content.handlebars',
|
|
|
|
'static/templates/admin_user_list.handlebars',
|
|
|
|
'static/templates/bot_avatar_row.handlebars',
|
|
|
|
'static/templates/draft.handlebars',
|
|
|
|
'static/templates/message_edit_form.handlebars',
|
|
|
|
'static/templates/message_edit_history.handlebars',
|
|
|
|
'static/templates/recipient_row.handlebars',
|
|
|
|
'static/templates/single_message.handlebars',
|
|
|
|
'static/templates/stream_privacy.handlebars',
|
|
|
|
'static/templates/stream_sidebar_row.handlebars',
|
|
|
|
'static/templates/subscription_settings.handlebars',
|
|
|
|
'static/templates/user_sidebar_actions.handlebars',
|
|
|
|
'static/templates/settings/account-settings.handlebars',
|
|
|
|
'static/templates/settings/admin-alias-list.handlebars',
|
|
|
|
'static/templates/settings/admin_auth_methods_list.handlebars',
|
|
|
|
'static/templates/settings/alert-word-settings.handlebars',
|
|
|
|
'static/templates/settings/attachment-item.handlebars',
|
|
|
|
'static/templates/settings/auth-methods-settings-admin.handlebars',
|
|
|
|
'static/templates/settings/bot-list-admin.handlebars',
|
|
|
|
'static/templates/settings/bot-settings.handlebars',
|
|
|
|
'static/templates/settings/deactivated-users-admin.handlebars',
|
|
|
|
'static/templates/settings/deactivation-stream-modal.handlebars',
|
|
|
|
'static/templates/settings/deactivation-user-modal.handlebars',
|
|
|
|
'static/templates/settings/default-streams-list-admin.handlebars',
|
|
|
|
'static/templates/settings/display-settings.handlebars',
|
|
|
|
'static/templates/settings/emoji-settings-admin.handlebars',
|
|
|
|
'static/templates/settings/muted-topics-settings.handlebars',
|
|
|
|
'static/templates/settings/notification-settings.handlebars',
|
|
|
|
'static/templates/settings/organization-settings-admin.handlebars',
|
|
|
|
'static/templates/settings/realm-domains-modal.handlebars',
|
|
|
|
'static/templates/settings/realm-filter-settings-admin.handlebars',
|
|
|
|
'static/templates/settings/streams-list-admin.handlebars',
|
|
|
|
'static/templates/settings/ui-settings.handlebars',
|
|
|
|
'static/templates/settings/user-list-admin.handlebars',
|
|
|
|
]
|
|
|
|
# TODO: Clean these files
|
|
|
|
for fn in templates:
|
|
|
|
if fn not in IGNORE_FILES:
|
|
|
|
if not validate_indent_html(fn):
|
|
|
|
sys.exit(1)
|
|
|
|
|
2014-02-27 16:41:47 +01:00
|
|
|
if __name__ == '__main__':
|
|
|
|
check_our_files()
|