Отправить письмо, написанное в уценке, используя mutt

21

Иногда мне нужно отправить фрагмент кода в google-группу inline. Текст здесь не помогает; Я могу напечатать его в markdown, преобразовать в html (используя pandoc и т. Д.), Прикрепить к Mutt as text/htmlи отправить.

Существует одно решение хорошо доступен здесь , но он использует внешнюю sendmailпрограмму для отправки электронной почты. Я использую Mutt, который имеет возможность отправлять электронные письма через IMAP самостоятельно.

Дилавар
источник
1
Почему бы просто не передать данные из форматера Markdown из командной строки в sendmail?
Нафтули Кей
Хм .. общий компьютер! Не хочу хранить пароль для внешнего sendmail.
Dilawar
Есть ли какие-нибудь примеры того, как мы можем видеть ваши текущие результаты в google-группах?
СЛМ
Кроме того, вы хотите напечатать вещи в уценке, но они были отображены до прикрепления их к вашим электронным письмам, верно?
СЛМ
Это звучит как то, что вы просите, но, возможно, потребуется изменить: dgl.cx/2009/03/html-mail-with-mutt-using-markdown . Также markdownmail.py звучало как то, что вы могли бы использовать.
СЛМ

Ответы:

28

После того, как вы написали сообщение, но перед отправкой у вас есть много вариантов, доступных для вас. Нажмите, ?чтобы просмотреть их.

Некоторые, которые могут помочь здесь:

  • F фильтровать вложение через внешний процессор
    • Используйте pandoc -s -f markdown -t htmlдля преобразования в HTML
  • ^T редактировать MIME тип вложения
    • Изменить с text/plainна text/html.

Теперь макрос, который сделает все за один шаг. Добавьте это к вашему .muttrc:

macro compose \e5 "F pandoc -s -f markdown -t html \ny^T^Utext/html; charset=us-ascii\n"
set wait_key=no

Чтобы использовать этот макрос, после того, как вы завершили создание своего сообщения, но перед отправкой, нажмите Escзатем, 5чтобы преобразовать ваше сообщение в формате уценки в HTML.

Вы можете естественным образом настроить этот макрос по своему усмотрению. В Mutt уже встроено множество привязок клавиш, поэтому, какую бы последовательность клавиш вы не выбрали, убедитесь, что она не перезаписывает что-то другое (или то, без чего вы можете жить).


Опция set wait_key=noподавляет Press any key to continue...запрос Mutt при выполнении внешних команд. Если wait_keyесть yes(это значение по умолчанию) , вы должны нажать Esc, а затем 5, а затем любую другую клавишу для продолжения.

bahamat
источник
1
Это действительно элегантное решение! +1
зловещий
5
это хорошо, но у него есть основной недостаток. это исключает текстовую часть электронного письма, что делает его неприемлемым для чтения в таких клиентах, как ... mutt;) В электронных письмах HTML должен быть компонент с открытым текстом и html. необработанная уценка должна быть открытым текстом, преобразованный должен быть HTML.
Масукоми
1
Согласитесь с @masukomi, почтовые клиенты в целом отправляют как html, так и текстовые версии письма. Было бы неплохо иметь макрос, который добавляет HTML-версию и оставляет оригинал в виде text / plain.
pepper_chico
2
В конце концов, я создаю свою собственную настройку для этого nosubstance.me/post/mutt-secret-sauce
pepper_chico
1

Sendmail часто не достаточно гибок для отправки писем.

Я использую msmtp вместе с Mutt на определенных учетных записях для гибкого SMTP.

Чтобы использовать его с изменениями Mutt:

# ~/.muttrc  
set sendmail="/usr/bin/msmtp -a default"   

а также

# ~/.msmtprc  
defaults
tls off
logfile ~/.msmtp.log  
account default   
host your.smtp.host  
port 25  
from your-user-name@your-host.com  
auth off  
user username  
password password  

источник
0

Я был в состоянии сделать это. Я не совсем доволен своим решением, но оно достаточно прилично. В ожидании кого-то, чтобы предоставить лучшее решение.

Процесс следующий. Преобразовать уценку в HTML и прикрепить его к сообщению. Превратите это вложение во inlineвложение. Но теперь у меня есть два вложения, первое в уценке, а второе HTML. Замените содержимое уценки пустой строкой, чтобы отправлялся только HTML.

Я добавил следующую строку в ~/.muttrcфайл.

macro compose B ":set editor=text2mime-markdown.py<enter>E:set editor=email-editor<enter>Da/tmp/html-markdown-alternative.html<enter>^Du"

Вот email-editorчто заимствовано из ссылки, размещенной в вопросе.

#!/bin/sh
if grep -q In-Reply-To $1; then
  # Jump to first line of message
  exec vim -c 'norm }j' $1
else
  # Enter insert mode on the To: line
  exec vim  $1
fi

И основной файл Python, который называется следующим. Это связано с Perl-скриптом из рассматриваемой ссылки.

#!/usr/bin/env python
import os
import sys
from formatter import *
version = "0.1"

file = sys.argv[1]
new_file = "/tmp/html-markdown-alternative.html"
with open(file, "r") as f:
    text = f.read()

lines = text.split('\n')
header = []
body = []
headerStart = True
for l in lines:
    if headerStart:
        m = re.search(r'^[\w\-]+\:', l)
        if m:
            header.append(l)
        else:
            headerStart = False
            body.append(l)
    else:
        body.append(l)

header = '\n'.join(header)
body = '\n'.join(body)

htmlBody = markdownToHtml(body);

html = []
html.append('<html>')
html.append('<head>')
html.append('<meta name=\"generator\" content=\"text2mime-markdown{}\">'.format(version))
html.append('<style>')
html.append("code { font-family: 'Andale Mono', 'Lucida Console', \
        'Bitstream Vera Sans Mono', 'Courier New', monospace; }")
html.append('pre { border-left: 20px solid #ddd; margin-left: 10px; \
        padding-left: 5px; }')
html.append('</style>')
html.append('</head>')
html.append('<body>')
html.append('{0}'.format(body))
html.append('</body>')
html.append('</html>')
html = '\n'.join(html)

with open(new_file, "w") as newF:
    newF.write(html)

with open(file, 'w') as f:
    f.write(header)

Это зависит от еще одного файла с именем python, formatter.pyкоторый используется pandocдля форматирования моей почты, но если pandocон недоступен, он может использовать python-markdown2пакет. Этот скрипт следующий.

import subprocess
import re
import os 
import sys
import html2text 
import collections

# check if pandoc exists
panDoc = True
try:
    subprocess.call(["pandoc", '--version']
            , stdout=subprocess.PIPE
            , stdin=subprocess.PIPE
            )
except OSError:
    panDoc = False

if not panDoc:
    import text.html2text as html2text
    import markdown 

def decodeText(text):
    return text.decode('utf-8')

def markdownToHtml(content, convertor='pandoc'):
    global panDoc
    if panDoc:
        cmd = ["pandoc", "-f", "markdown", "-t", "html"]
        p = subprocess.Popen(cmd
                , stdin = subprocess.PIPE
                , stdout = subprocess.PIPE
                )
        p.stdin.write(content)
        content = p.communicate()[0]
        return decodeText(content)
    else:
        return markdown.markdown(decodeText(content))


def htmlToMarkdown(content, convertor='pandoc'):
    global panDoc
    if panDoc and convertor == 'pandoc':
        cmd = ["pandoc", "-t", "markdown", "-f", "html"]
        p = subprocess.Popen(cmd
                , stdin = subprocess.PIPE
                , stdout = subprocess.PIPE
                )
        p.stdin.write(content)
        content = p.communicate()[0]
        return decodeText(content)
    # Use markdown package to convert markdown to html
    else:
        h = html2text.HTML2Text()
        content = h.handle(decodeText(content))
        return content

def titleToBlogDir(title):
    if title is None:
        return ''
    if len(title.strip()) == 0:
        return ''
    blogDir = title.replace(" ","_").replace(':', '-').replace('(', '')
    blogDir = blogDir.replace('/', '').replace('\\', '').replace('`', '')
    blogDir = blogDir.replace(')', '').replace("'", '').replace('"', '')
    return blogDir

def titleToFilePath(title, blogDir):
    if len(blogDir.strip()) == 0:
        return ''
    fileName = os.path.join(blogDir, titleToBlogDir(title))
    return fileName


def htmlToHtml(html):
    return decodeText(html)

def metadataDict(txt):
    mdict = collections.defaultdict(list)
    md = getMetadata(txt)
    for c in ["title", 'type', "layout", "status", "id", "published", "category", "tag"]:
        pat = re.compile(r'{0}\:\s*(?P<name>.+)'.format(c), re.IGNORECASE)
        m = pat.findall(txt)
        for i in m:
            mdict[c].append(i)
    return mdict

def getMetadata(txt):
   """
   Get metadata out of a txt
   """
   if not "---" in txt:
       print txt
       sys.exit(1)

   pat = re.compile(r'\-\-\-+(?P<metadata>.+?)\-\-\-+', re.DOTALL)
   m = pat.search(txt)
   if m:
       return m.group('metadata')
   else:
       sys.exit(0)

def getContent(txt):
    """ 
    Return only text of the post.
    """
    pat = re.compile(r'\-\-\-+(?P<metadata>.+?)\-\-\-+', re.DOTALL)
    return re.sub(pat, "", txt)

def readInputFile(fileName):
    """
    read file and return its format. html or markdown
    """
    assert fileName
    if not os.path.exists(fileName):
        raise IOError, "File %s does not exists" % fileName

    # Check the fmt of file.
    fmt = os.path.splitext(fileName)[1].lower()
    if fmt in ["htm", "html", "xhtml"]:
        fmt = "html"
    elif fmt in ["md", "markdown"]:
        fmt = "markdown"
    else:
        fmt = "markdown"
    txt = open(fileName, 'r').read()   
    return (fmt, txt)

def formatContent(txt, fmt):
    """
    Format the content as per fmt.
    """
    content = getContent(txt)
    if fmt == "html":
        content = htmlToHtml(content)
    elif fmt == "markdown":
        content = markdownToHtml(content)
    else:
        content = markdownToHtml(content)
    return content

Эти файлы также доступны здесь https://github.com/dilawar/mutt

Дилавар
источник
0

Я могу отправить письмо в любом формате, используя neomutt. Я просто использую Emacs(org-mode) вместо vim. Хотя я тоже vimпользователь. Но я в основном использую Emacsзлой режим.

По моему .muttrcя настроил редактор emacsвместо vim. При написании нового электронного письма neomuttзапускается emacs. Затем я вызываю «org-mode», пишу сообщение и экспортирую в любой нужный мне формат.

Я могу экспортировать в PDFформат. Затем я сохраняю его и прикрепляю PDFфайл к моему /tmp. После этого я могу отправить кому угодно.

Если мне нужен htmlформат, я экспортирую его таким же образом, и я автоматически вижу результат перед отправкой электронного письма.

Кроме этого, в org-mode есть много других форматов экспорта. Просто выберите то, что вы хотите. Для отправки кода другим людям просто добавьте исходный код на любой язык, который вы хотите. Все объясняется в org-wiki .

Achylles
источник
0

Вы можете отправлять электронные письма также как multipart/alternativeсодержащие text/plainи text/html.

Требования: Пандок

В основном это создает из открытого текста сообщения уценки и html5. Создает вложения из этих частей, помечает их как встроенные вложения, устанавливает правильный тип MIME и объединяет их в многопользовательское сообщение.

Любые другие вложения должны быть добавлены после запуска этого макроса в меню создания. Факультативное подписание / шифрование сообщения должно быть сделано в качестве последнего шага

macro compose ,m \
"<enter-command>set pipe_decode<enter>\
<pipe-message>pandoc -f gfm -t plain -o /tmp/msg.txt<enter>\
<pipe-message>pandoc -s -f gfm -t html5 -o /tmp/msg.html<enter>\
<enter-command>unset pipe_decode<enter>a^U/tmp/msg.txt\n^Da^U/tmp/msg.html\n^D^T^Utext/html; charset=utf-8\n=DTT&d^U\n" \
"Convert markdown gfm to HTML and plain" 
Якуб Джиндра
источник