Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement function to prune string keeping HTML closing tags #1870

Open
wants to merge 7 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
120 changes: 120 additions & 0 deletions backend/app/notify/prune_html.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
package notify

import (
"fmt"
"strings"

"golang.org/x/net/html"
)

type stringArr struct {
data []string
len int
}

// Push adds element to the end
func (s *stringArr) Push(v string) {
s.data = append(s.data, v)
s.len += len(v)
}

// Pop removes element from end and returns it
func (s *stringArr) Pop() string {
l := len(s.data)
newData, v := s.data[:l-1], s.data[l-1]
s.data = newData
s.len -= len(v)
return v
}

// Unshift adds element to the start
func (s *stringArr) Unshift(v string) {
s.data = append([]string{v}, s.data...)
s.len += len(v)
}

// Shift removes element from start and returns it
func (s *stringArr) Shift() string {
v, newData := s.data[0], s.data[1:]
s.data = newData
s.len -= len(v)
return v
}

// String returns all strings concatenated
func (s stringArr) String() string {
return strings.Join(s.data, "")
}

// Len returns total length of all strings concatenated
func (s stringArr) Len() int {
return s.len
}

// pruneHTML prunes string keeping HTML closing tags
func pruneHTML(htmlText string, maxLength int) string {
result := stringArr{}
endTokens := stringArr{}

suffix := "..."
suffixLen := len(suffix)

tokenizer := html.NewTokenizer(strings.NewReader(htmlText))
for {
if tokenizer.Next() == html.ErrorToken {
return result.String()
}
token := tokenizer.Token()

switch token.Type {
case html.CommentToken, html.DoctypeToken:
// skip tokens without content
continue

case html.StartTagToken:
// <token></token>
// len(token) * 2 + len("<></>")
totalLenToAppend := len(token.Data)*2 + 5

lengthAfterChange := result.Len() + totalLenToAppend + endTokens.Len() + suffixLen

if lengthAfterChange > maxLength {
return result.String() + suffix + endTokens.String()
}

endTokens.Unshift(fmt.Sprintf("</%s>", token.Data))

case html.EndTagToken:
endTokens.Shift()

case html.TextToken, html.SelfClosingTagToken:
lengthAfterChange := result.Len() + len(token.String()) + endTokens.Len() + suffixLen

if lengthAfterChange > maxLength {
text := pruneStringToWord(token.String(), maxLength-result.Len()-endTokens.Len()-suffixLen)
return result.String() + text + suffix + endTokens.String()
}
}

result.Push((token.String()))
}
}

// pruneStringToWord prunes string to specified length respecting words
func pruneStringToWord(text string, maxLength int) string {
if maxLength <= 0 {
return ""
}

result := ""

arr := strings.Split(text, " ")
for _, s := range arr {
if len(result)+len(s) >= maxLength {
return strings.TrimRight(result, " ")
}
result += s + " "
}

return text
}
4 changes: 3 additions & 1 deletion backend/app/notify/telegram.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ import (
"github.com/hashicorp/go-multierror"
)

const commentTextLengthLimit = 100

// TelegramParams contain settings for telegram notifications
type TelegramParams struct {
AdminChannelID string // unique identifier for the target chat or username of the target channel (in the format @channelusername)
Expand Down Expand Up @@ -85,7 +87,7 @@ func (t *Telegram) buildMessage(req Request) string {
msg += fmt.Sprintf(" -> <a href=%q>%s</a>", commentURLPrefix+req.parent.ID, ntf.EscapeTelegramText(req.parent.User.Name))
}

msg += fmt.Sprintf("\n\n%s", ntf.TelegramSupportedHTML(req.Comment.Text))
msg += fmt.Sprintf("\n\n%s", pruneHTML(ntf.TelegramSupportedHTML(req.Comment.Text), commentTextLengthLimit))

if req.Comment.ParentID != "" {
msg += fmt.Sprintf("\n\n\"<i>%s</i>\"", ntf.TelegramSupportedHTML(req.parent.Text))
Expand Down
9 changes: 9 additions & 0 deletions backend/app/notify/telegram_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,15 @@ some text

<b>Hello</b><i><b>World</b></i>`,
res)

// prune string keeping HTML closing tags
c = store.Comment{
Text: "<b>Lorem ipsum <i>dolor sit amet</i>, consectetur adipiscing <code>elit, sed do eiusmod tempor incididunt</code> ut labore et dolore magna aliqua.</b>",
}
res = tb.buildMessage(Request{Comment: c})
assert.Equal(t, `<a href="#remark42__comment-"></a>

<b>Lorem ipsum <i>dolor sit amet</i>, consectetur adipiscing <code>elit, sed do...</code></b>`, res)
}

func TestTelegram_SendVerification(t *testing.T) {
Expand Down
Loading