import asyncdispatch, smtp, strutils, json, os, rst, rstgen, xmltree, strtabs, htmlparser, streams, parseutils, options from times import getTime, getGMTime, format # Used to be: # {'A'..'Z', 'a'..'z', '0'..'9', '_', '\128'..'\255'} let UsernameIdent* = IdentChars # TODO: Double check that everyone follows this. import frontend/karaxutils export parseInt proc `%`*[T](opt: Option[T]): JsonNode = ## Generic constructor for JSON data. Creates a new ``JNull JsonNode`` ## if ``opt`` is empty, otherwise it delegates to the underlying value. if opt.isSome: %opt.get else: newJNull() type Config* = object smtpAddress: string smtpPort: int smtpUser: string smtpPassword: string mlistAddress: string recaptchaSecretKey*: string recaptchaSiteKey*: string var docConfig: StringTableRef docConfig = rstgen.defaultConfig() docConfig["doc.listing_start"] = "
"
docConfig["doc.listing_end"] = "
" proc loadConfig*(filename = getCurrentDir() / "forum.json"): Config = result = Config(smtpAddress: "", smtpPort: 25, smtpUser: "", smtpPassword: "", mlistAddress: "") try: let root = parseFile(filename) result.smtpAddress = root{"smtpAddress"}.getStr("") result.smtpPort = root{"smtpPort"}.getNum(25).int result.smtpUser = root{"smtpUser"}.getStr("") result.smtpPassword = root{"smtpPassword"}.getStr("") result.mlistAddress = root{"mlistAddress"}.getStr("") result.recaptchaSecretKey = root{"recaptchaSecretKey"}.getStr("") result.recaptchaSiteKey = root{"recaptchaSiteKey"}.getStr("") except: echo("[WARNING] Couldn't read config file: ", filename) proc processGT(n: XmlNode, tag: string): (int, XmlNode, string) = result = (0, newElement(tag), tag) if n.kind == xnElement and len(n) == 1 and n[0].kind == xnElement: return processGT(n[0], if n[0].kind == xnElement: n[0].tag else: tag) var countGT = true for c in items(n): case c.kind of xnText: if c.text == ">" and countGT: result[0].inc() else: countGT = false result[1].add(newText(c.text)) else: result[1].add(c) proc blockquoteFinish(currentBlockquote, newNode: var XmlNode, n: XmlNode) = if currentBlockquote.len > 0: #echo(currentBlockquote.repr) newNode.add(currentBlockquote) currentBlockquote = newElement("blockquote") newNode.add(n) proc processQuotes(node: XmlNode): XmlNode = # Bolt on quotes. # TODO: Yes, this is ugly. I wrote it quickly. PRs welcome ;) result = newElement("div") var currentBlockquote = newElement("blockquote") for n in items(node): case n.kind of xnElement: case n.tag of "p": let (nesting, contentNode, _) = processGT(n, "p") if nesting > 0: var bq = currentBlockquote for i in 1 ..< nesting: var newBq = bq.child("blockquote") if newBq.isNil: newBq = newElement("blockquote") bq.add(newBq) bq = newBq bq.add(contentNode) else: blockquoteFinish(currentBlockquote, result, n) else: blockquoteFinish(currentBlockquote, result, n) of xnText: if n.text[0] == '\10': result.add(n) else: blockquoteFinish(currentBlockquote, result, n) else: blockquoteFinish(currentBlockquote, result, n) proc replaceMentions(node: XmlNode): seq[XmlNode] = assert node.kind == xnText result = @[] var current = "" var i = 0 while i < len(node.text): i += parseUntil(node.text, current, {'@'}, i) if i >= len(node.text): break if node.text[i] == '@': i.inc # Skip @ var username = "" i += parseWhile(node.text, username, UsernameIdent, i) if username.len == 0: result.add(newText(current & "@")) else: let el = <>span( class="user-mention", data-username=username, newText("@" & username) ) result.add(newText(current)) current = "" result.add(el) result.add(newText(current)) proc processMentions(node: XmlNode): XmlNode = case node.kind of xnText: result = newElement("span") for child in replaceMentions(node): result.add(child) of xnElement: case node.tag of "pre", "code", "tt": return node else: result = newElement(node.tag) result.attrs = node.attrs for n in items(node): result.add(processMentions(n)) else: return node proc rstToHtml*(content: string): string = result = rstgen.rstToHtml(content, {roSupportMarkdown}, docConfig) try: var node = parseHtml(newStringStream(result)) if node.kind == xnElement: node = processQuotes(node) node = processMentions(node) result = "" add(result, node, indWidth=0, addNewLines=false) except: echo("[WARNING] Could not parse rst html.") proc sendMail(config: Config, subject, message, recipient: string, from_addr = "forum@nim-lang.org", otherHeaders:seq[(string, string)] = @[]) {.async.} = if config.smtpAddress.len == 0: echo("[WARNING] Cannot send mail: no smtp server configured (smtpAddress).") return var client = newAsyncSmtp() await client.connect(config.smtpAddress, Port(config.smtpPort)) if config.smtpUser.len > 0: await client.auth(config.smtpUser, config.smtpPassword) let toList = @[recipient] var headers = otherHeaders headers.add(("From", from_addr)) let encoded = createMessage(subject, message, toList, @[], headers) await client.sendMail(from_addr, toList, $encoded) proc sendMailToMailingList*(config: Config, username, user_email_addr, subject, message: string, threadUrl: string, thread_id=0, post_id=0, is_reply=false) {.async.} = # send message to a mailing list if config.mlistAddress.len == 0: echo("[WARNING] Cannot send mail: no mlistAddress configured.") return let from_addr = "$# <$#>" % [username, user_email_addr] let date = getTime().getGMTime().format("ddd, d MMM yyyy HH:mm:ss") & " +0000" var otherHeaders = @[ ("Date", date), ("Resent-From", "forum@nim-lang.org"), ("Resent-date", date) ] if is_reply: let msg_id = "" % [$thread_id, $post_id] otherHeaders.add(("Message-ID", msg_id)) let references = "" % [$thread_id] otherHeaders.add(("References", references)) else: # New thread let msg_id = "" % $thread_id otherHeaders.add(("Message-ID", msg_id)) var processedMsg: string try: processedMsg = rstToHTML(message) & "
View thread on Nim forum" otherHeaders.add(("Content-Type", "text/html; charset=\"UTF-8\"")) except: processedMsg = message await sendMail(config, subject, processedMsg, config.mlistAddress, from_addr=from_addr, otherHeaders=otherHeaders) proc sendPassReset*(config: Config, email, user, resetUrl: string) {.async.} = let message = """Hello $1, A password reset has been requested for your account on the Nim Forum. If you did not make this request, you can safely ignore this email. A password reset request can be made by anyone, and it does not indicate that your account is in any danger of being accessed by someone else. If you do actually want to reset your password, visit this link: $2 Thank you for being a part of the Nim community!""" % [user, resetUrl] await sendMail(config, "Nim Forum Password Recovery", message, email) proc sendEmailActivation*(config: Config, email, user, activateUrl: string) {.async.} = let message = """Hello $1, You have recently registered an account on the Nim Forum. As the final step in your registration, we require that you confirm your email via the following link: $2 Thank you for registering and becoming a part of the Nim community!""" % [user, activateUrl] await sendMail(config, "Nim Forum Account Email Confirmation", message, email)