fabfile.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. #!/usr/bin/env python
  2. from __future__ import print_function
  3. ##### Configuration ##############################
  4. SHORT_PROJECT_NAME = 'python'
  5. FULL_PROJECT_NAME = 'byte_of_{}'.format(SHORT_PROJECT_NAME)
  6. # NOTE Slugs MUST be lower-case
  7. MARKDOWN_FILES = [
  8. {
  9. 'file': '01-frontpage.md',
  10. 'slug': "python",
  11. 'title': "Python",
  12. },
  13. {
  14. 'file': '02-preface.md',
  15. 'slug': "python_en-preface",
  16. 'title': "Preface",
  17. },
  18. {
  19. 'file': '03-intro.md',
  20. 'slug': "python_en-introduction",
  21. 'title': "Introduction",
  22. },
  23. {
  24. 'file': '04-installation.md',
  25. 'slug': "python_en-installation",
  26. 'title': "Installation",
  27. },
  28. ]
  29. ## NOTES
  30. ## 1. This assumes that you have already created the S3 bucket whose name
  31. ## is stored in AWS_S3_BUCKET_NAME environment variable.
  32. ## 2. Under that S3 bucket, you have created a folder whose name is stored
  33. ## above as SHORT_PROJECT_NAME.
  34. ## 3. Under that S3 bucket, you have created a folder whose name is stored as
  35. ## SHORT_PROJECT_NAME/assets.
  36. ##### Imports ####################################
  37. import os
  38. import glob
  39. import subprocess
  40. try:
  41. from xmlrpc.client import ServerProxy
  42. except ImportError:
  43. from xmlrpclib import ServerProxy
  44. from pprint import pprint
  45. import boto
  46. import boto.s3.bucket
  47. import boto.s3.key
  48. from bs4 import BeautifulSoup
  49. from fabric.api import task, local
  50. ##### Start with checks ##########################
  51. for chapter in MARKDOWN_FILES:
  52. assert (chapter['slug'].lower() == chapter['slug']), \
  53. "Slug must be lower case : {}".format(chapter['slug'])
  54. if str(os.environ.get('AWS_ENABLED')).lower() == 'false':
  55. AWS_ENABLED = False
  56. elif os.environ.get('AWS_ACCESS_KEY_ID') is not None \
  57. and len(os.environ['AWS_ACCESS_KEY_ID']) > 0 \
  58. and os.environ.get('AWS_SECRET_ACCESS_KEY') is not None \
  59. and len(os.environ['AWS_SECRET_ACCESS_KEY']) > 0 \
  60. and os.environ.get('AWS_S3_BUCKET_NAME') is not None \
  61. and len(os.environ['AWS_S3_BUCKET_NAME']) > 0:
  62. AWS_ENABLED = True
  63. else:
  64. AWS_ENABLED = False
  65. print("NOTE: S3 uploading is disabled because of missing " +
  66. "AWS key environment variables.")
  67. # In my case, they are the same - 'files.swaroopch.com'
  68. # http://docs.amazonwebservices.com/AmazonS3/latest/dev/VirtualHosting.html#VirtualHostingCustomURLs
  69. S3_PUBLIC_URL = os.environ['AWS_S3_BUCKET_NAME']
  70. # else
  71. #S3_PUBLIC_URL = 's3.amazonaws.com/{}'.format(os.environ['AWS_S3_BUCKET_NAME'])
  72. if os.environ.get('WORDPRESS_RPC_URL') is not None \
  73. and len(os.environ['WORDPRESS_RPC_URL']) > 0 \
  74. and os.environ.get('WORDPRESS_BASE_URL') is not None \
  75. and len(os.environ['WORDPRESS_BASE_URL']) > 0 \
  76. and os.environ.get('WORDPRESS_BLOG_ID') is not None \
  77. and len(os.environ['WORDPRESS_BLOG_ID']) > 0 \
  78. and os.environ.get('WORDPRESS_USERNAME') is not None \
  79. and len(os.environ['WORDPRESS_USERNAME']) > 0 \
  80. and os.environ.get('WORDPRESS_PASSWORD') is not None \
  81. and len(os.environ['WORDPRESS_PASSWORD']) > 0 \
  82. and os.environ.get('WORDPRESS_PARENT_PAGE_ID') is not None \
  83. and len(os.environ['WORDPRESS_PARENT_PAGE_ID']) > 0 \
  84. and os.environ.get('WORDPRESS_PARENT_PAGE_SLUG') is not None \
  85. and len(os.environ['WORDPRESS_PARENT_PAGE_SLUG']) > 0:
  86. WORDPRESS_ENABLED = True
  87. else:
  88. WORDPRESS_ENABLED = False
  89. print("NOTE: Wordpress uploading is disabled because of " +
  90. "missing environment variables.")
  91. ##### Helper methods #############################
  92. def _upload_to_s3(filename, key):
  93. """http://docs.pythonboto.org/en/latest/s3_tut.html#storing-data"""
  94. conn = boto.connect_s3()
  95. b = boto.s3.bucket.Bucket(conn, os.environ['AWS_S3_BUCKET_NAME'])
  96. k = boto.s3.key.Key(b)
  97. k.key = key
  98. k.set_contents_from_filename(filename)
  99. k.set_acl('public-read')
  100. url = 'http://{}/{}'.format(S3_PUBLIC_URL, key)
  101. print("Uploaded to S3 : {}".format(url))
  102. return url
  103. def upload_output_to_s3(filename):
  104. key = "{}/{}".format(SHORT_PROJECT_NAME, filename.split('/')[-1])
  105. return _upload_to_s3(filename, key)
  106. def upload_asset_to_s3(filename):
  107. key = "{}/assets/{}".format(SHORT_PROJECT_NAME, filename.split('/')[-1])
  108. return _upload_to_s3(filename, key)
  109. def replace_images_with_s3_urls(text):
  110. """http://www.crummy.com/software/BeautifulSoup/bs4/doc/"""
  111. soup = BeautifulSoup(text)
  112. for image in soup.find_all('img'):
  113. image['src'] = upload_asset_to_s3(image['src'])
  114. return soup.prettify()
  115. def markdown_to_html(source_text, upload_assets_to_s3=False):
  116. """Convert from Markdown to HTML; optional: upload images, etc. to S3."""
  117. args = ['pandoc',
  118. '-f', 'markdown',
  119. '-t', 'html5',
  120. '-S']
  121. p = subprocess.Popen(args, stdin=subprocess.PIPE, stdout=subprocess.PIPE)
  122. output = p.communicate(source_text)[0]
  123. if upload_assets_to_s3:
  124. output = replace_images_with_s3_urls(output)
  125. return output
  126. def _wordpress_get_pages():
  127. server = ServerProxy(os.environ['WORDPRESS_RPC_URL'])
  128. print("(Fetching list of pages from WP)")
  129. return server.wp.getPosts(os.environ['WORDPRESS_BLOG_ID'],
  130. os.environ['WORDPRESS_USERNAME'],
  131. os.environ['WORDPRESS_PASSWORD'],
  132. {
  133. 'post_type': 'page',
  134. 'number': pow(10, 5),
  135. })
  136. def wordpress_new_page(slug, title, content):
  137. """Create a new Wordpress page.
  138. https://codex.wordpress.org/XML-RPC_WordPress_API/Posts#wp.newPost
  139. https://codex.wordpress.org/Function_Reference/wp_insert_post
  140. http://docs.python.org/library/xmlrpclib.html
  141. """
  142. server = ServerProxy(os.environ['WORDPRESS_RPC_URL'])
  143. return server.wp.newPost(os.environ['WORDPRESS_BLOG_ID'],
  144. os.environ['WORDPRESS_USERNAME'],
  145. os.environ['WORDPRESS_PASSWORD'],
  146. {
  147. 'post_name': slug,
  148. 'post_content': content,
  149. 'post_title': title,
  150. 'post_parent':
  151. os.environ['WORDPRESS_PARENT_PAGE_ID'],
  152. 'post_type': 'page',
  153. 'post_status': 'publish',
  154. 'comment_status': 'closed',
  155. 'ping_status': 'closed',
  156. })
  157. def wordpress_edit_page(post_id, content):
  158. """Edit a Wordpress page.
  159. https://codex.wordpress.org/XML-RPC_WordPress_API/Posts#wp.editPost
  160. https://codex.wordpress.org/Function_Reference/wp_insert_post
  161. http://docs.python.org/library/xmlrpclib.html
  162. """
  163. server = ServerProxy(os.environ['WORDPRESS_RPC_URL'])
  164. return server.wp.editPost(os.environ['WORDPRESS_BLOG_ID'],
  165. os.environ['WORDPRESS_USERNAME'],
  166. os.environ['WORDPRESS_PASSWORD'],
  167. post_id,
  168. {
  169. 'post_content': content,
  170. })
  171. ##### Tasks ######################################
  172. @task
  173. def wp():
  174. """https://codex.wordpress.org/XML-RPC_WordPress_API/Posts"""
  175. if WORDPRESS_ENABLED:
  176. existing_pages = _wordpress_get_pages()
  177. existing_page_slugs = [i.get('post_name') for i in existing_pages]
  178. def page_slug_to_id(slug):
  179. pages = [i for i in existing_pages if i.get('post_name') == slug]
  180. page = pages[0]
  181. return page['post_id']
  182. for chapter in MARKDOWN_FILES:
  183. html = markdown_to_html(open(chapter['file']).read(),
  184. upload_assets_to_s3=True)
  185. # TODO Add previous and next links at end of html
  186. if chapter['slug'] in existing_page_slugs:
  187. page_id = page_slug_to_id(chapter['slug'])
  188. print("Existing page to be updated: {} : {}".format(
  189. chapter['slug'],
  190. page_id))
  191. result = wordpress_edit_page(page_id, html)
  192. print("Result: {}".format(result))
  193. else:
  194. print("New page to be created: {}".format(chapter['slug']))
  195. result = wordpress_new_page(chapter['slug'],
  196. chapter['title'],
  197. html)
  198. print("Result: {}".format(result))
  199. page_url = "{}/{}/{}".format(os.environ['WORDPRESS_BASE_URL'],
  200. os.environ['WORDPRESS_PARENT_PAGE_SLUG'],
  201. chapter['slug'])
  202. print(page_url)
  203. print()
  204. @task
  205. def html():
  206. """HTML5 output."""
  207. args = ['pandoc',
  208. '-f', 'markdown',
  209. '-t', 'html5',
  210. '-o', '{}.html'.format(FULL_PROJECT_NAME),
  211. '-S',
  212. '-s',
  213. '--toc'] + [i['file'] for i in MARKDOWN_FILES]
  214. local(' '.join(args))
  215. local('open {}.html'.format(FULL_PROJECT_NAME))
  216. @task
  217. def epub():
  218. """http://johnmacfarlane.net/pandoc/epub.html"""
  219. args = ['pandoc',
  220. '-f', 'markdown',
  221. '-t', 'epub',
  222. '-o', '{}.epub'.format(FULL_PROJECT_NAME),
  223. '-S'] + [i['file'] for i in MARKDOWN_FILES]
  224. # TODO --epub-cover-image
  225. # TODO --epub-metadata
  226. # TODO --epub-stylesheet
  227. local(' '.join(args))
  228. if AWS_ENABLED:
  229. upload_output_to_s3('{}.epub'.format(FULL_PROJECT_NAME))
  230. @task
  231. def pdf():
  232. """http://johnmacfarlane.net/pandoc/README.html#creating-a-pdf"""
  233. args = ['pandoc',
  234. '-f', 'markdown',
  235. # https://github.com/jgm/pandoc/issues/571
  236. #'-t', 'pdf',
  237. '-o', '{}.pdf'.format(FULL_PROJECT_NAME),
  238. '-S'] + [i['file'] for i in MARKDOWN_FILES]
  239. local(' '.join(args))
  240. if AWS_ENABLED:
  241. upload_output_to_s3('{}.pdf'.format(FULL_PROJECT_NAME))
  242. @task
  243. def docx():
  244. """OOXML document format."""
  245. args = ['pandoc',
  246. '-f', 'markdown',
  247. '-t', 'docx',
  248. '-o', '{}.docx'.format(FULL_PROJECT_NAME),
  249. '-S'] + [i['file'] for i in MARKDOWN_FILES]
  250. local(' '.join(args))
  251. if AWS_ENABLED:
  252. upload_output_to_s3('{}.docx'.format(FULL_PROJECT_NAME))
  253. @task
  254. def odt():
  255. """OpenDocument document format."""
  256. args = ['pandoc',
  257. '-f', 'markdown',
  258. '-t', 'odt',
  259. '-o', '{}.odt'.format(FULL_PROJECT_NAME),
  260. '-S'] + [i['file'] for i in MARKDOWN_FILES]
  261. local(' '.join(args))
  262. if AWS_ENABLED:
  263. upload_output_to_s3('{}.odt'.format(FULL_PROJECT_NAME))
  264. @task
  265. def clean():
  266. """Remove generated output files"""
  267. possible_outputs = (
  268. '{}.html'.format(FULL_PROJECT_NAME),
  269. '{}.epub'.format(FULL_PROJECT_NAME),
  270. '{}.pdf'.format(FULL_PROJECT_NAME),
  271. '{}.docx'.format(FULL_PROJECT_NAME),
  272. '{}.odt'.format(FULL_PROJECT_NAME),
  273. )
  274. for filename in possible_outputs:
  275. if os.path.exists(filename):
  276. os.remove(filename)
  277. print("Removed {}".format(filename))