2020-12-21 13:12:05 +01:00
# lib_src2md.rb
2021-01-11 03:15:04 +01:00
require 'pathname'
2020-12-21 13:12:05 +01:00
2021-01-11 03:15:04 +01:00
# The method 'src2md' convert .src.md file into .md file.
# The output .md file is fit for the final format, which is one of markdown, html and latex.
# - Links to relative URL are removed for latex. Otherwise, it remains.
# See "Hyperref and relative link" below for further explanation.
# - Width and height for images are removed for markdown and html. it remains for latex.
# ![sample](sample_image){width=10cm height=5cm} => ![sample](sample_image) for markdown and html
# ---- Hyperref and relative link ----
# Hyperref package makes internal link possible.
# The target of the link is made with '\hypertarget' command.
# And the link is made with '\hyperlink' command.
# For example,
# (sec11.tex)
# \hyperlink{tfeapplication.c}{Section 13}
# ... ...
# (sec13.tex)
# \hypertarget{tfeapplication.c}{%
# \section{tfeapplication.c}\label{tfeapplication.c}}
2021-02-08 14:24:54 +01:00
# If you click the text 'Section 13' in sec11.tex, then you can move to '13 tfeapplication.c' ("13 " is automatically added by latex), which is section 13 in sec13.tex.
2021-01-11 03:15:04 +01:00
# The following lines are the original one in sec11.md and the result in sec11.tex, which is generated by pandoc.
# (sec11.md)
# All the source files are listed in [Section 13](sec13.tex).
# (sec11.tex)
# All the source files are listed in \href{sec13.tex}{Section 13}.
# Therefore, if you want to correct the link in sec11.tex, you need to do the followings.
# 1. Look at the first line of sec13.md and get the section heading (tfeapplication.c).
# 2. substitute "\hyperlink{tfeapplication.c}{Section 13}" for "\href{sec13.tex}{Section 13}".
# The following lines are another conversion case by pandoc.
# (sec7.md)
# The source code of `tfe3.c` is stored in [src/tfe](../src/tfe) directory.
# (sec7.tex)
# The source code of \texttt{tfe3.c} is stored in \href{../src/tfe}{src/tfe} directory.
# The pdf file generated by pdflatex recognizes that the link 'href{../src/tfe}' points a pdf file '../src/tfe.pdf'.
# To avoid generating such incorrect links, it is good to remove the links from the original markdown file.
# If the target is full URL, which means absolute URL begins with "http", no problem happens.
2021-02-08 14:24:54 +01:00
# This script just remove the links if its target is relative URL.
2021-01-11 03:15:04 +01:00
# If you want to revive the link with relative URL, refer the description above.
# ---- Folding verbatim lines ----
# When C sourcefiles or subshell output are included, the lines are folded to fit in 'width'.
2021-01-25 10:35:49 +01:00
# Width must be positive integer.
2021-01-11 03:15:04 +01:00
# Otherwise the lines are not folded.
2021-01-25 10:35:49 +01:00
# This script uses "fenced code blocks" for verbatim lines.
# It is available in GFM and pandoc's markdown but not in original markdown.
# Two characters backtick (`) and tilde (~) are possible for fences.
# This script uses tilde because info string cannot contain any backticks for the backtick code fence.
# Info string follows opening fence and it is usually a language name.
# ~~~C
# int main (int argc, char **argv) {
# ........
# ~~~
# Then the contents are highlighted based on C language syntax.
# This script find the language by the suffix of the file name.
# .c => C, .h => C, .rb => ruby, Rakefile, => ruby, .xml => xml, .ui => xml, .y => bison, .lex => lex, .build => meson, .md => markdown
# Makefile => makefile
2021-01-11 03:15:04 +01:00
def src2md srcmd , md , width
2021-02-08 14:24:54 +01:00
# parameters:
# srcmd: .src.md file's path. source
# md: .md file's path. destination
# width: maximum width of lines in fence code block
2020-12-21 13:12:05 +01:00
src_buf = IO . readlines srcmd
src_dir = File . dirname srcmd
2021-01-11 03:15:04 +01:00
md_dir = File . dirname md
2021-02-08 14:24:54 +01:00
type = File . basename md_dir # type of the target. gfm, html or latex
2021-01-11 03:15:04 +01:00
2020-12-21 13:12:05 +01:00
md_buf = [ ]
2021-02-08 14:24:54 +01:00
include_flag = " "
shell_flag = false
if_stat = 0
2020-12-21 13:12:05 +01:00
src_buf . each do | line |
2021-02-08 14:24:54 +01:00
if include_flag == " -N " || include_flag == " -n "
if line == " @@@ \n "
include_flag = false
elsif line =~ / ^ *( \ S*) *(.*)$ /
c_file = $1
c_functions = $2 . strip . split ( " " )
if c_file =~ / ^ \/ / # absolute path
c_file_buf = IO . readlines ( c_file )
else #relative path
c_file_buf = IO . readlines ( src_dir + " / " + c_file )
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
if c_functions . empty? # no functions are specified
tmp_buf = c_file_buf
else
tmp_buf = [ ]
spc = false
c_functions . each do | c_function |
from = c_file_buf . find_index { | line | line =~ / ^ #{ c_function } * \ ( / }
if ! from
warn " ERROR!!! --- Didn't find #{ c_function } in #{ filename } . --- "
2020-12-21 13:12:05 +01:00
break
end
2021-02-08 14:24:54 +01:00
to = from
while to < c_file_buf . size do
if c_file_buf [ to ] == " } \n "
break
end
to += 1
end
n = from - 1
if spc
tmp_buf << " \n "
else
spc = true
end
while n < = to do
tmp_buf << c_file_buf [ n ]
n += 1
end
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
end
md_buf << " ~~~ #{ lang ( c_file ) } \n "
ln_width = tmp_buf . size . to_s . length
n = 1
tmp_buf . each do | l |
if include_flag == " -n "
l = sprintf ( " % #{ ln_width } d %s " , n , l )
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
fold ( l , width ) . each_line do | l2 |
md_buf << l2
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
n += 1
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
md_buf << " ~~~ \n "
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
elsif shell_flag
if line == " @@@ \n "
shell_flag = false
else
md_buf << " ~~~ \n "
fold ( " $ #{ line } " , width ) . each_line do | l2 |
2021-01-25 10:35:49 +01:00
md_buf << l2
end
2021-02-08 14:24:54 +01:00
` cd #{ src_dir } ; #{ line . chomp } ` . each_line do | l |
fold ( l , width ) . each_line do | l2 |
md_buf << l2
end
end
md_buf << " ~~~ \n "
2020-12-21 13:12:05 +01:00
end
2021-02-08 14:24:54 +01:00
elsif line =~ / ^@@@if *( \ w+) / && if_stat == 0
if_stat = type == $1 ? 1 : - 1
elsif line =~ / ^@@@elif *( \ w+) /
if if_stat == 1
if_stat = - 2
elsif if_stat == - 1
if_stat = type == $1 ? 3 : - 3
elsif if_stat == - 2
# if_stat is kept to be -2
elsif if_stat == 3
if_stat = - 2
elsif if_stat == - 3
if_stat = type == $1 ? 3 : - 3
end
elsif line =~ / ^@@@else /
if if_stat == 1
if_stat = - 2
elsif if_stat == - 1
if_stat = 2
elsif if_stat == - 2
# if_stat is kept to be -2
elsif if_stat == 3
if_stat = - 2
elsif if_stat == - 3
if_stat = 2
end
elsif line =~ / ^@@@end /
if_stat = 0
elsif if_stat > = 0
if line == " @@@include \n " || line =~ / ^@@@include *-n /
include_flag = " -n "
elsif line =~ / ^@@@include *-N /
include_flag = " -N "
elsif line == " @@@shell \n "
shell_flag = true
else
line = change_rel_link ( line , src_dir , md_dir )
if type == " latex " # remove relative link
line . gsub! ( / (^|[^!]) \ [([^ \ ]]*) \ ] \ ((?~http) \ ) / , " \\ 1 \\ 2 " )
else # type == "gfm" or "html", then remove size option from link to image files.
line . gsub! ( / (! \ [[^ \ ]]* \ ] \ ([^ \ )]* \ )) *{width *= * \ d*(| \ . \ d*)cm *height *= * \ d*(| \ . \ d*)cm} / , " \\ 1 " )
end
md_buf << line
2021-01-11 03:15:04 +01:00
end
end
end
2020-12-21 13:12:05 +01:00
IO . write ( md , md_buf . join )
end
2021-01-25 10:35:49 +01:00
# Change the base of relative links from org_dir to new_dir
def change_rel_link line , org_dir , new_dir
p_new_dir = Pathname . new new_dir
2021-01-11 03:15:04 +01:00
left = " "
right = line
2021-01-24 08:50:52 +01:00
while right =~ / (!? \ [[^ \ ]]* \ ]) \ (([^ \ )]*) \ ) /
2021-02-06 09:26:57 +01:00
left += $`
2021-01-11 03:15:04 +01:00
right = $'
name = $1
link = $2
if name =~ / \ [(S|s)ection ( \ d+) \ ] /
link = " sec #{ $2 } .md "
elsif ! ( link =~ / ^(http| \/ ) / )
2021-01-25 10:35:49 +01:00
p_link = Pathname . new " #{ org_dir } / #{ link } "
link = p_link . relative_path_from ( p_new_dir ) . to_s
2021-01-11 03:15:04 +01:00
end
left += " #{ name } ( #{ link } ) "
end
left + right
end
2021-01-25 10:35:49 +01:00
def fold line , width
2021-02-08 14:24:54 +01:00
if line . instance_of? ( String ) && width . instance_of? ( Integer ) && width > 0
n = ( line . chomp . length - 1 ) / width
n . downto ( 1 ) do | i |
line . insert width * i , " \n "
end
2021-01-25 10:35:49 +01:00
end
2021-02-08 14:24:54 +01:00
line
2021-01-25 10:35:49 +01:00
end
def lang file
tbl = { " .c " = > " C " , " .h " = > " C " , " .rb " = > " ruby " , " .xml " = > " xml " , " .ui " = > " xml " ,
" .y " = > " bison " , " .lex " = > " lex " , " .build " = > " meson " , " .md " = > " markdown " }
name = File . basename file
if name == " Makefile "
return " makefile "
elsif name == " Rakefile "
return " ruby "
else
suffix = File . extname name
tbl . each do | key , val |
if suffix == key
return val
end
end
end
return " "
end