2018-09-15 06:17:04 +02:00
|
|
|
import re
|
2020-10-20 01:28:13 +02:00
|
|
|
from typing import Any, List, Optional
|
2018-09-15 06:17:04 +02:00
|
|
|
from typing.re import Match
|
2020-06-11 00:54:34 +02:00
|
|
|
|
2020-10-19 06:37:43 +02:00
|
|
|
from markdown import Markdown
|
|
|
|
from markdown.extensions import Extension
|
2018-09-15 06:17:04 +02:00
|
|
|
from markdown.preprocessors import Preprocessor
|
|
|
|
|
|
|
|
# There is a lot of duplicated code between this file and
|
|
|
|
# help_settings_links.py. So if you're making a change here consider making
|
|
|
|
# it there as well.
|
|
|
|
|
|
|
|
REGEXP = re.compile(r'\{relative\|(?P<link_type>.*?)\|(?P<key>.*?)\}')
|
|
|
|
|
|
|
|
gear_info = {
|
|
|
|
# The pattern is key: [name, link]
|
|
|
|
# key is from REGEXP: `{relative|gear|key}`
|
|
|
|
# name is what the item is called in the gear menu: `Select **name**.`
|
|
|
|
# link is used for relative links: `Select [name](link).`
|
|
|
|
'manage-streams': ['Manage streams', '/#streams/subscribed'],
|
|
|
|
'settings': ['Settings', '/#settings/your-account'],
|
|
|
|
'manage-organization': ['Manage organization', '/#organization/organization-profile'],
|
|
|
|
'integrations': ['Integrations', '/integrations'],
|
|
|
|
'stats': ['Statistics', '/stats'],
|
|
|
|
'plans': ['Plans and pricing', '/plans'],
|
|
|
|
'billing': ['Billing', '/billing'],
|
2018-09-16 16:06:16 +02:00
|
|
|
'invite': ['Invite users', '/#invite'],
|
2018-09-15 06:17:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gear_instructions = """
|
|
|
|
1. From your desktop, click on the **gear**
|
2018-10-19 01:12:15 +02:00
|
|
|
(<i class="fa fa-cog"></i>) in the upper right corner.
|
2018-09-15 06:17:04 +02:00
|
|
|
|
2020-07-10 01:57:43 +02:00
|
|
|
1. Select {item}.
|
2018-09-15 06:17:04 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
def gear_handle_match(key: str) -> str:
|
|
|
|
if relative_help_links:
|
2020-06-10 06:41:04 +02:00
|
|
|
item = f'[{gear_info[key][0]}]({gear_info[key][1]})'
|
2018-09-15 06:17:04 +02:00
|
|
|
else:
|
2020-06-10 06:41:04 +02:00
|
|
|
item = f'**{gear_info[key][0]}**'
|
2020-07-10 01:57:43 +02:00
|
|
|
return gear_instructions.format(item=item)
|
2018-09-15 06:17:04 +02:00
|
|
|
|
2018-09-16 04:29:56 +02:00
|
|
|
|
|
|
|
stream_info = {
|
|
|
|
'all': ['All streams', '/#streams/all'],
|
|
|
|
'subscribed': ['Your streams', '/#streams/subscribed'],
|
|
|
|
}
|
|
|
|
|
|
|
|
stream_instructions_no_link = """
|
|
|
|
1. From your desktop, click on the **gear**
|
2018-10-19 01:12:15 +02:00
|
|
|
(<i class="fa fa-cog"></i>) in the upper right corner.
|
2018-09-16 04:29:56 +02:00
|
|
|
|
|
|
|
1. Click **Manage streams**.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def stream_handle_match(key: str) -> str:
|
|
|
|
if relative_help_links:
|
2020-06-10 06:41:04 +02:00
|
|
|
return f"1. Go to [{stream_info[key][0]}]({stream_info[key][1]})."
|
2018-09-16 04:29:56 +02:00
|
|
|
if key == 'all':
|
|
|
|
return stream_instructions_no_link + "\n\n1. Click **All streams** in the upper left."
|
|
|
|
return stream_instructions_no_link
|
|
|
|
|
|
|
|
|
2018-09-15 06:17:04 +02:00
|
|
|
LINK_TYPE_HANDLERS = {
|
|
|
|
'gear': gear_handle_match,
|
2018-09-16 04:29:56 +02:00
|
|
|
'stream': stream_handle_match,
|
2018-09-15 06:17:04 +02:00
|
|
|
}
|
|
|
|
|
2020-10-19 06:37:43 +02:00
|
|
|
class RelativeLinksHelpExtension(Extension):
|
|
|
|
def extendMarkdown(self, md: Markdown) -> None:
|
2018-09-15 06:17:04 +02:00
|
|
|
""" Add RelativeLinksHelpExtension to the Markdown instance. """
|
|
|
|
md.registerExtension(self)
|
2020-10-20 01:28:13 +02:00
|
|
|
md.preprocessors.register(RelativeLinks(), 'help_relative_links', 520)
|
2018-09-15 06:17:04 +02:00
|
|
|
|
python: Convert assignment type annotations to Python 3.6 style.
This commit was split by tabbott; this piece covers the vast majority
of files in Zulip, but excludes scripts/, tools/, and puppet/ to help
ensure we at least show the right error messages for Xenial systems.
We can likely further refine the remaining pieces with some testing.
Generated by com2ann, with whitespace fixes and various manual fixes
for runtime issues:
- invoiced_through: Optional[LicenseLedger] = models.ForeignKey(
+ invoiced_through: Optional["LicenseLedger"] = models.ForeignKey(
-_apns_client: Optional[APNsClient] = None
+_apns_client: Optional["APNsClient"] = None
- notifications_stream: Optional[Stream] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
- signup_notifications_stream: Optional[Stream] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
+ notifications_stream: Optional["Stream"] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
+ signup_notifications_stream: Optional["Stream"] = models.ForeignKey('Stream', related_name='+', null=True, blank=True, on_delete=CASCADE)
- author: Optional[UserProfile] = models.ForeignKey('UserProfile', blank=True, null=True, on_delete=CASCADE)
+ author: Optional["UserProfile"] = models.ForeignKey('UserProfile', blank=True, null=True, on_delete=CASCADE)
- bot_owner: Optional[UserProfile] = models.ForeignKey('self', null=True, on_delete=models.SET_NULL)
+ bot_owner: Optional["UserProfile"] = models.ForeignKey('self', null=True, on_delete=models.SET_NULL)
- default_sending_stream: Optional[Stream] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
- default_events_register_stream: Optional[Stream] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
+ default_sending_stream: Optional["Stream"] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
+ default_events_register_stream: Optional["Stream"] = models.ForeignKey('zerver.Stream', null=True, related_name='+', on_delete=CASCADE)
-descriptors_by_handler_id: Dict[int, ClientDescriptor] = {}
+descriptors_by_handler_id: Dict[int, "ClientDescriptor"] = {}
-worker_classes: Dict[str, Type[QueueProcessingWorker]] = {}
-queues: Dict[str, Dict[str, Type[QueueProcessingWorker]]] = {}
+worker_classes: Dict[str, Type["QueueProcessingWorker"]] = {}
+queues: Dict[str, Dict[str, Type["QueueProcessingWorker"]]] = {}
-AUTH_LDAP_REVERSE_EMAIL_SEARCH: Optional[LDAPSearch] = None
+AUTH_LDAP_REVERSE_EMAIL_SEARCH: Optional["LDAPSearch"] = None
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-04-22 01:09:50 +02:00
|
|
|
relative_help_links: Optional[bool] = None
|
2018-09-15 06:17:04 +02:00
|
|
|
|
|
|
|
def set_relative_help_links(value: bool) -> None:
|
|
|
|
global relative_help_links
|
|
|
|
relative_help_links = value
|
|
|
|
|
|
|
|
class RelativeLinks(Preprocessor):
|
|
|
|
def run(self, lines: List[str]) -> List[str]:
|
|
|
|
done = False
|
|
|
|
while not done:
|
|
|
|
for line in lines:
|
|
|
|
loc = lines.index(line)
|
|
|
|
match = REGEXP.search(line)
|
|
|
|
|
|
|
|
if match:
|
|
|
|
text = [self.handleMatch(match)]
|
|
|
|
# The line that contains the directive to include the macro
|
|
|
|
# may be preceded or followed by text or tags, in that case
|
|
|
|
# we need to make sure that any preceding or following text
|
|
|
|
# stays the same.
|
|
|
|
line_split = REGEXP.split(line, maxsplit=0)
|
|
|
|
preceding = line_split[0]
|
|
|
|
following = line_split[-1]
|
2020-09-02 06:59:07 +02:00
|
|
|
text = [preceding, *text, following]
|
2018-09-15 06:17:04 +02:00
|
|
|
lines = lines[:loc] + text + lines[loc+1:]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
done = True
|
|
|
|
return lines
|
|
|
|
|
|
|
|
def handleMatch(self, match: Match[str]) -> str:
|
|
|
|
return LINK_TYPE_HANDLERS[match.group('link_type')](match.group('key'))
|
|
|
|
|
|
|
|
def makeExtension(*args: Any, **kwargs: Any) -> RelativeLinksHelpExtension:
|
|
|
|
return RelativeLinksHelpExtension(*args, **kwargs)
|