Markdown转HTML之Node篇
前言
之前用Python写过类似的工具,更能上来说一般般。而且实用性不是很强。http://blog.csdn.net/marksinoberg/article/details/51863506
然后好巧不巧又看到了一个Node上的相关模块,看起来渲染效果比我那个好多了。联想到hexo这款静态博客生成工具,就思量着也大致的做一下。
环境及编码
接下来简单的模拟一下相关的实现。代码比较凌乱,思维比较混乱,还望海涵(虽然主要是作为我自己的笔记,但如果对你有些许帮助,那就不枉我码了这么多字咯)。
搭建环境
所依赖的第三方模块有如下几个:
express
: 开启本地服务, 预览生成效果。Markdown-it
: 渲染md文件为HTML内容。rd
: 一个读取文件夹内容的好帮手。commander
: 制作命令行工具的一大利器。
下面简要对这几个模块进行阐述,以及相关的使用技巧。
express
express不仅作为对connect的高层封装,更包含了一些额外的处理。所以我们可以方便的进行路由控制,这对于本次的工具而言,是个不错的选择。
// 初始化服务器
var app = express();
var router = express.Router();
app.use('/assets', server_static(path.resolve(dir, 'assets')));
app.use(router);
// 渲染文章
router.get('/posts/*', function(req, res, next){
var name = stripExtname(req.params[0]);
// 渲染req.params[0]对应的文章,然后展示给前台
res.end(html);
});
// res.end(req.params[0]);
});
// 渲染列表
router.get('/', function(req, res, next){
// 读取源文件目录, 渲染出列表内容。
res.end('list of articles.');
});
app.listen(8080);
markdown-it
相对于Python中的那些第三方库,我倒是觉得Node中与其也没甚么两样。使用起来同样很简单。
第一步,引入依赖
let markdowner = require('markdown-it');
第二步, 配置构造器
var md = new markdowner({
html: true,
prefix: 'code-',
});
第三步, 调用渲染方法,获取渲染后内容
var html = md.render(sourcedata||'');
如此,便是markdown-it的基础内容了,待会将在代码中更加详细的运用。
commander
用过Python的argparser的估计都知道,很方便的一个处理命令行参数的第三方库。不过Node中的commander用起来更方便。
下面简单的介绍一下使用流程。详细内容还是看人家的官网吧,如下:
https://www.npmjs.com/package/commander
第一步, 安装模块
npm install commander --save
第二步,编码
/**
* 一个命令行工具库。
*/
let commander = require('commander');
// help 命令
commander.command('help')
.description('显示工具如何使用的帮助信息')
.action(function(){
commander.outputHelp();
});
// create 命令
commander.command('create [dirname]')
.description('创建一个空的博客')
.action(function(dirname){
console.log(dirname+' 创建完成。')
});
// preview 命令
commander.command('preview [dirname]')
.description('预览获取到的Markdown文件夹内容')
.action(function(dirname){
console.log(' preview of %s', dirname);
});
// .action(require('./cmd_preview'));
// build 命令
commander.command('build [dirname]')
.description('根据给定的文件夹路径生成HTML内容.')
.option('-o OR --output <dirname>', '导出生成的HTML存放的路径')
.action(function(dirname){
console.log('build based on %s', dirname);
});
// .action(require('./cmd_build'));
// 解析相关命令
commander.parse(process.argv);
第三步, 查看效果
如果想更加方便一点,直接使用命令来操作。使用
npm link
当然了,还需要设置一下对应的package.json
文件内容。这里不过多叙述了。
rd
我这里的需求是读取_posts文件夹下的Markdown源文件,所以只需要readFile方法即可。
具体代码如下:
rd.readFile(sourcedir, function(err, files){
if(err){
console.log('读取文件夹内容失败!');
return;
}
// 遍历文件夹列表,对每一个文件执行渲染操作。
files.forEach(function(file){
// 做自己的逻辑处理即可。
});
});
这个模块比较简单,有兴趣的可以参考下面的作者链接。
https://github.com/leizongmin/node-rd
核心编码
下面正式开始今天的主题,做一个带预览功能的Markdown文件转HTML页面的工具。
cmd_preview模块
/**
* 关于预览实现相关的代码。
*/
let express = require('express');
let path = require('path');
let markdowner = require('markdown-it');
let fs = require('fs');
let rd = require('rd');
var md = new markdowner({
html: true,
langPrefix: 'code-',
});
module.exports = function(dir) {
dir = dir || '.';
// 初始化服务器
var app = express();
var router = express.Router();
app.use(router);
// 渲染文章
router.get('/posts/*', function(req, res, next){
var name = stripExtname(req.params[0]);
var file = path.resolve(dir, '_posts', name+'.md');
console.log('---dir--', dir);
console.log('---name--', name);
console.log('---file--', file);
fs.readFile(file, function(err, content){
if(err){
console.log('读取文件失败!');
res.end(JSON.stringify(err)+"\n");
return next(err);
}
res.writeHead(200, {"Content-Type": "text/html;charset=UTF-8"});
var html = markdownTOHTML(content.toString());
console.log('读取文件成功, 解析后的内容为:\n', html);
res.end(html);
});
// res.end(req.params[0]);
});
// 渲染列表
router.get('/', function(req, res, next){
var sourcefolder = path.resolve(dir, '_posts');
rd.readFile(sourcefolder, function(err, files){
if(err){
console.log('读取文件夹内文件失败!');
return next(err);
}
res.writeHead(200, {"Content-Type": "text/html;charset=UTF-8"});
var html = "<html><h1>Markdown 转 HTML 实时预览</h1><hr><br />";
files.forEach(function(filepath){
html += "<a href='/posts/"+ get_file_name(filepath) +".md' target='_blank'>"+get_file_name(filepath)+"</a><br /><br />";
});
html += "</html>";
res.end(html);
});
// res.end('list of articles.');
});
app.listen(8080);
};
function stripExtname(name) {
var i = 0-path.extname(name).length;
if(i==0) i=name.length;
return name.slice(0, i);
}
function get_file_name(fullname){
var ls = fullname.toString().split('\\');
var filename = ls[ls.length-1].split('.');
// console.log('ls--', ls);
// console.log('filename--', filename);
return filename[0];
}
function markdownTOHTML(content) {
return md.render(content||'');
}
cmd_build模块
/**
* 实现Markdown文件到HTML文件的转换。
*/
let markdowner = require('markdown-it');
let rd = require('rd');
let path = require('path');
let fs = require('fs');
var md = new markdowner({
html: true,
langPrefix: 'code-',
})
module.exports = function(dir) {
dir = dir || '.';
console.log('当前文件路径为:', dir);
// 读取出给定目录下的所有的文件
// 将所有Markdown文件依次转成HTML页面,并进行保存操作。
get_files_by_dir(dir);
}
function get_files_by_dir(dir) {
// 计算出源文件的路径
var sourcedir = path.resolve(dir, '_posts');
var publicdir = path.resolve(dir, 'public');
rd.readFile(sourcedir, function(err, files){
if(err){
console.log('读取文件夹内容失败!');
return;
}
// 遍历文件夹列表,对每一个文件执行渲染操作。
files.forEach(function(file){
var html = md2html(file);
var filename = get_filename_by_path(file);
var output = path.resolve(publicdir, filename+'.html');
console.log('保存路径为:', output);
save_html_content(html, output);
console.log('%s.html 生成成功!', filename);
});
});
}
function md2html(filepath){
var content = fs.readFileSync(filepath);
var html = md.render(content.toString()||'');
return "<html><head><meta charset='UTF-8'><title>"+get_filename_by_path(filepath)+"</title></head>"+html+"</html>";
}
function save_html_content(content, outpath){
fs.writeFile(outpath, content, function(err){
if(err){
console.log('save_html_content: 保存文件内容失败!');
return;
}
console.log('save_html_content: %s 保存成功!', outpath);
});
}
function get_filename_by_path(filepath){
var paths = filepath.toString().split('\\');
return paths[paths.length-1].split('.')[0];
}
打造命令行工具
还是用刚才的hello.js,现在稍微修改一下action里面的内容,对应我们刚才做的那两个小模块,做下修改即可。完整代码如下:
/**
* 一个命令行工具库。
*/
let commander = require('commander');
// help 命令
commander.command('help')
.description('显示工具如何使用的帮助信息')
.action(function(){
commander.outputHelp();
});
// create 命令
commander.command('create [dirname]')
.description('创建一个空的博客')
.action(function(dirname){
console.log(dirname+' 创建完成。')
});
// preview 命令
commander.command('preview [dirname]')
.description('预览获取到的Markdown文件夹内容')
// .action(function(dirname){
// console.log(' preview of %s', dirname);
// });
.action(require('./cmd_preview'));
// build 命令
commander.command('build [dirname]')
.description('根据给定的文件夹路径生成HTML内容.')
.option('-o OR --output <dirname>', '导出生成的HTML存放的路径')
// .action(function(dirname){
// console.log('build based on %s', dirname);
// });
.action(require('./cmd_build'));
// 解析相关命令
commander.parse(process.argv);
写点xx.md
巧妇难为无米之炊, 现在先在hello.js的同级目录下建个文件夹_posts,里面写点xx.md文件,然后建一个public文件夹保存生成的HTML文件。比如我的目录结构是这样的。
E:\Code\Nodejs\learn\libs-learn\commander-related>tree /f .
卷 文档 的文件夹 PATH 列表
卷序列号为 0000-4823
E:\CODE\NODEJS\LEARN\LIBS-LEARN\COMMANDER-RELATED
│ cmd_build.js
│ cmd_preview.js
│ hello.js
│
├─public
│
└─_posts
helloworld.md
second.md
演示
首先是预览实现,在命令行里输入以下命令:
node hello.js preview . # .代表当前目录
结果如下:
然后是针对每一篇文章生成效果的演示。
最后就是看下build功能的实现。
现在在命令行里面输入以下命令:
node hello.js build . # .代表当前目录
然后看下效果。
总结
大致来说功能就算是完成了,但是目前这样子是没法直接应用的。很多东西都需要润色,比如:
- 模板化HTML页面处理。
- 命令行选项的link实现。
- 通过监测文件内容变化实现实时预览。
就先到这里吧,今天又发现了两个不错的网址,然后还是先去学一波,补充补充知识吧。很多时候,不是能力不够,而是见识不足。
Markdown转HTML之Node篇的更多相关文章
- C蛮的全栈之路-node篇(二) 实战一:自动发博客
目录 C蛮的全栈之路-序章 技术栈选择与全栈工程师C蛮的全栈之路-node篇(一) 环境布置C蛮的全栈之路-node篇(二) 实战一:自动发博客 ---------------- 我是分割线 ---- ...
- C蛮的全栈之路-node篇(一) 环境布置
目录 C蛮的全栈之路-序章 技术栈选择与全栈工程师C蛮的全栈之路-node篇(一) 环境布置C蛮的全栈之路-node篇(二) 实战一:自动发博客 ---------------- 我是分割线 ---- ...
- Node篇
[Node篇] Node.js中的stream(流)- 基础篇 1)什么是stream(流) 流(stream)在 Node.js 中是处理流数据的抽象接口(abstract interface). ...
- 细说WebSocket - Node篇
在上一篇提高到了 web 通信的各种方式,包括 轮询.长连接 以及各种 HTML5 中提到的手段.本文将详细描述 WebSocket协议 在 web通讯 中的实现. 一.WebSocket 协议 1. ...
- 学习OpenStack之(6):Neutron 深入学习之 OVS + GRE 之 Compute node 篇
0.环境 硬件环境见上一篇博客:学习OpenStack之(5):在Mac上部署Juno版本OpenStack 四节点环境 OpenStack网络配置:一个tenant, 2个虚机 Type drive ...
- 基于 Markdown 的开源的 Node.js 知识库平台
Raneto 是一个免费,开源的 Node.js 知识库平台,基于静态 Markdown 文件实现. Raneto 可以被称为静态网站生成器,因为它并不需要数据库支持.所有的内容都存储在 Markdo ...
- K8S入门系列之集群二进制部署-->node篇(三)
node节点组件 docker kubelet kube-proxy kubernetes-server-linux-amd64.tar.gz(相关的这里都能找到二进制文件!) falnnel 1. ...
- 前端面试题整理—Node篇
1.node有哪些特征,与其他服务器端对比 特征:单线程.事件驱动.非阻塞I/O node 无法直接渲染静态页面,提供静态服务 node 没有根目录的概念 node 必须通过路由程序指定文件才能渲染文 ...
- 【Node/JavaScript】论一个低配版Web实时通信库是如何实现的( WebSocket篇)
引论 simple-socket是我写的一个"低配版"的Web实时通信工具(相对于Socket.io),在参考了相关源码和资料的基础上,实现了前后端实时互通的基本功能 选用了Web ...
随机推荐
- ruby项目文件上传功能实现
这里我将从视图.控制器各个层面进行讲解. rails 提供了文件上传功能,可以直接进行下面的编码 <%= form_for :document, :html =>{:multipart = ...
- LINQ 获取当前数组中出现次数最多的元素
LINQ 获取当前数组中出现次数最多的元素 1 List<string> a = new List<string>(); a.Add( ...
- sqlserver导入excel的电话号码(身份证)变为科学计数解决方式
如果excel中有一列存的是手机号码或者身份证号码,那么导入到sql中时,会把手机或者身份证当作数字格式对待,因而会以科学记数法的形式存在sqlserver表中,解决方式,先将excel文件另存为文本 ...
- selet2使用大全
selet2是一款input+selet结合的组件,是最好的搜索下拉框,没有之一.原因是别的没有,只有它这么一家,你没得选择.正如有人说它‘Select2不是特别好用,但又找不到比它更好的下拉框插件. ...
- HDU - 3829 Cat VS Dog (二分图最大独立集)
题意:P个小朋友,每个人有喜欢的动物和讨厌的动物.留下喜欢的动物并且拿掉讨厌的动物,这个小朋友就会开心.问最多有几个小朋友能开心. 分析:对于每个动物来说,可能既有人喜欢又有人讨厌,那么这样的动物实际 ...
- hadoop20---代理另一种方式
package cn.itcast_05_proxy.service; /** * 这是一个业务的接口,这个接口中的业务就是返回衣服的价格 */ public interface IBoss {//接 ...
- Nginx URL跳转
配置需求 注:$document_uri 表示访问的url 需求:访问 www.abc.com 请求到 www.abc.com/abc/ 使用操作 1.在nginx配置文件中加入 if ($doc ...
- 20145219 《Java程序设计》第08周学习总结
20145219 <Java程序设计>第08周学习总结 教材学习内容总结 通用API 日志API 1.java.util.logging包提供了日志功能相关类与接口,使用日志的起点是log ...
- python的正则表达式捕获组命名问题
- 如何自定义echarts 线性图的选择事件
最近在做公司的数据大盘,要用到图表显示,echarts研二的时候有用过,我就决定用它了. 这里用到一个可以同时显示多条曲线的line-charts,基本样子如下: 看到这个画红色圈圈的地方了吗??? ...