You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

426 lines
17KB

  1. extern crate site;
  2. extern crate tempdir;
  3. extern crate glob;
  4. use std::env;
  5. use std::path::Path;
  6. use std::fs::File;
  7. use std::io::prelude::*;
  8. use tempdir::TempDir;
  9. use site::Site;
  10. #[test]
  11. fn can_parse_site() {
  12. let mut path = env::current_dir().unwrap().to_path_buf();
  13. path.push("test_site");
  14. let mut site = Site::new(&path, "config.toml").unwrap();
  15. site.load().unwrap();
  16. // Correct number of pages (sections are pages too)
  17. assert_eq!(site.pages.len(), 11);
  18. let posts_path = path.join("content").join("posts");
  19. // Make sure we remove all the pwd + content from the sections
  20. let basic = &site.pages[&posts_path.join("simple.md")];
  21. assert_eq!(basic.file.components, vec!["posts".to_string()]);
  22. // Make sure the page with a url doesn't have any sections
  23. let url_post = &site.pages[&posts_path.join("fixed-url.md")];
  24. assert_eq!(url_post.path, "a-fixed-url/");
  25. // Make sure the article in a folder with only asset doesn't get counted as a section
  26. let asset_folder_post = &site.pages[&posts_path.join("with-assets").join("index.md")];
  27. assert_eq!(asset_folder_post.file.components, vec!["posts".to_string()]);
  28. // That we have the right number of sections
  29. assert_eq!(site.sections.len(), 6);
  30. // And that the sections are correct
  31. let index_section = &site.sections[&path.join("content").join("_index.md")];
  32. assert_eq!(index_section.subsections.len(), 2);
  33. assert_eq!(index_section.pages.len(), 1);
  34. let posts_section = &site.sections[&posts_path.join("_index.md")];
  35. assert_eq!(posts_section.subsections.len(), 1);
  36. assert_eq!(posts_section.pages.len(), 5);
  37. let tutorials_section = &site.sections[&posts_path.join("tutorials").join("_index.md")];
  38. assert_eq!(tutorials_section.subsections.len(), 2);
  39. assert_eq!(tutorials_section.pages.len(), 0);
  40. let devops_section = &site.sections[&posts_path.join("tutorials").join("devops").join("_index.md")];
  41. assert_eq!(devops_section.subsections.len(), 0);
  42. assert_eq!(devops_section.pages.len(), 2);
  43. let prog_section = &site.sections[&posts_path.join("tutorials").join("programming").join("_index.md")];
  44. assert_eq!(prog_section.subsections.len(), 0);
  45. assert_eq!(prog_section.pages.len(), 2);
  46. }
  47. // 2 helper macros to make all the build testing more bearable
  48. macro_rules! file_exists {
  49. ($root: expr, $path: expr) => {
  50. {
  51. let mut path = $root.clone();
  52. for component in $path.split("/") {
  53. path = path.join(component);
  54. }
  55. Path::new(&path).exists()
  56. }
  57. }
  58. }
  59. macro_rules! file_contains {
  60. ($root: expr, $path: expr, $text: expr) => {
  61. {
  62. let mut path = $root.clone();
  63. for component in $path.split("/") {
  64. path = path.join(component);
  65. }
  66. let mut file = File::open(&path).unwrap();
  67. let mut s = String::new();
  68. file.read_to_string(&mut s).unwrap();
  69. println!("{}", s);
  70. s.contains($text)
  71. }
  72. }
  73. }
  74. #[test]
  75. fn can_build_site_without_live_reload() {
  76. let mut path = env::current_dir().unwrap().to_path_buf();
  77. path.push("test_site");
  78. let mut site = Site::new(&path, "config.toml").unwrap();
  79. site.load().unwrap();
  80. let tmp_dir = TempDir::new("example").expect("create temp dir");
  81. let public = &tmp_dir.path().join("public");
  82. site.set_output_path(&public);
  83. site.build().unwrap();
  84. assert!(Path::new(&public).exists());
  85. assert!(file_exists!(public, "index.html"));
  86. assert!(file_exists!(public, "sitemap.xml"));
  87. assert!(file_exists!(public, "robots.txt"));
  88. assert!(file_exists!(public, "a-fixed-url/index.html"));
  89. assert!(file_exists!(public, "posts/python/index.html"));
  90. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  91. assert!(file_exists!(public, "posts/with-assets/index.html"));
  92. assert!(file_exists!(public, "posts/no-section/simple/index.html"));
  93. // Sections
  94. assert!(file_exists!(public, "posts/index.html"));
  95. assert!(file_exists!(public, "posts/tutorials/index.html"));
  96. assert!(file_exists!(public, "posts/tutorials/devops/index.html"));
  97. assert!(file_exists!(public, "posts/tutorials/programming/index.html"));
  98. // TODO: add assertion for syntax highlighting
  99. // aliases work
  100. assert!(file_exists!(public, "an-old-url/old-page/index.html"));
  101. assert!(file_contains!(public, "an-old-url/old-page/index.html", "something-else"));
  102. // redirect_to works
  103. assert!(file_exists!(public, "posts/tutorials/devops/index.html"));
  104. assert!(file_contains!(public, "posts/tutorials/devops/index.html", "docker"));
  105. // No tags or categories
  106. assert_eq!(file_exists!(public, "categories/index.html"), false);
  107. assert_eq!(file_exists!(public, "tags/index.html"), false);
  108. // Theme files are there
  109. assert!(file_exists!(public, "sample.css"));
  110. assert!(file_exists!(public, "some.js"));
  111. // no live reload code
  112. assert_eq!(file_contains!(public, "index.html", "/livereload.js?port=1112&mindelay=10"), false);
  113. // Both pages and sections are in the sitemap
  114. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/posts/simple/</loc>"));
  115. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/posts/</loc>"));
  116. }
  117. #[test]
  118. fn can_build_site_with_live_reload() {
  119. let mut path = env::current_dir().unwrap().to_path_buf();
  120. path.push("test_site");
  121. let mut site = Site::new(&path, "config.toml").unwrap();
  122. site.load().unwrap();
  123. let tmp_dir = TempDir::new("example").expect("create temp dir");
  124. let public = &tmp_dir.path().join("public");
  125. site.set_output_path(&public);
  126. site.enable_live_reload();
  127. site.build().unwrap();
  128. assert!(Path::new(&public).exists());
  129. assert!(file_exists!(public, "index.html"));
  130. assert!(file_exists!(public, "sitemap.xml"));
  131. assert!(file_exists!(public, "robots.txt"));
  132. assert!(file_exists!(public, "a-fixed-url/index.html"));
  133. assert!(file_exists!(public, "posts/python/index.html"));
  134. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  135. assert!(file_exists!(public, "posts/with-assets/index.html"));
  136. // Sections
  137. assert!(file_exists!(public, "posts/index.html"));
  138. assert!(file_exists!(public, "posts/tutorials/index.html"));
  139. assert!(file_exists!(public, "posts/tutorials/devops/index.html"));
  140. assert!(file_exists!(public, "posts/tutorials/programming/index.html"));
  141. // TODO: add assertion for syntax highlighting
  142. // No tags or categories
  143. assert_eq!(file_exists!(public, "categories/index.html"), false);
  144. assert_eq!(file_exists!(public, "tags/index.html"), false);
  145. // no live reload code
  146. assert!(file_contains!(public, "index.html", "/livereload.js?port=1112&mindelay=10"));
  147. }
  148. #[test]
  149. fn can_build_site_with_categories() {
  150. let mut path = env::current_dir().unwrap().to_path_buf();
  151. path.push("test_site");
  152. let mut site = Site::new(&path, "config.toml").unwrap();
  153. site.config.generate_categories_pages = Some(true);
  154. site.load().unwrap();
  155. for (i, page) in site.pages.values_mut().enumerate() {
  156. page.meta.category = if i % 2 == 0 {
  157. Some("A".to_string())
  158. } else {
  159. Some("B".to_string())
  160. };
  161. }
  162. site.populate_tags_and_categories();
  163. let tmp_dir = TempDir::new("example").expect("create temp dir");
  164. let public = &tmp_dir.path().join("public");
  165. site.set_output_path(&public);
  166. site.build().unwrap();
  167. assert!(Path::new(&public).exists());
  168. assert_eq!(site.categories.unwrap().len(), 2);
  169. assert!(file_exists!(public, "index.html"));
  170. assert!(file_exists!(public, "sitemap.xml"));
  171. assert!(file_exists!(public, "robots.txt"));
  172. assert!(file_exists!(public, "a-fixed-url/index.html"));
  173. assert!(file_exists!(public, "posts/python/index.html"));
  174. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  175. assert!(file_exists!(public, "posts/with-assets/index.html"));
  176. // Sections
  177. assert!(file_exists!(public, "posts/index.html"));
  178. assert!(file_exists!(public, "posts/tutorials/index.html"));
  179. assert!(file_exists!(public, "posts/tutorials/devops/index.html"));
  180. assert!(file_exists!(public, "posts/tutorials/programming/index.html"));
  181. // TODO: add assertion for syntax highlighting
  182. // Categories are there
  183. assert!(file_exists!(public, "categories/index.html"));
  184. assert!(file_exists!(public, "categories/a/index.html"));
  185. assert!(file_exists!(public, "categories/b/index.html"));
  186. // Extending from a theme works
  187. assert!(file_contains!(public, "categories/a/index.html", "EXTENDED"));
  188. // Tags aren't
  189. assert_eq!(file_exists!(public, "tags/index.html"), false);
  190. // Categories are in the sitemap
  191. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/categories/</loc>"));
  192. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/categories/a/</loc>"));
  193. }
  194. #[test]
  195. fn can_build_site_with_tags() {
  196. let mut path = env::current_dir().unwrap().to_path_buf();
  197. path.push("test_site");
  198. let mut site = Site::new(&path, "config.toml").unwrap();
  199. site.config.generate_tags_pages = Some(true);
  200. site.load().unwrap();
  201. for (i, page) in site.pages.values_mut().enumerate() {
  202. page.meta.tags = if i % 2 == 0 {
  203. Some(vec!["tag1".to_string(), "tag2".to_string()])
  204. } else {
  205. Some(vec!["tag with space".to_string()])
  206. };
  207. }
  208. site.populate_tags_and_categories();
  209. let tmp_dir = TempDir::new("example").expect("create temp dir");
  210. let public = &tmp_dir.path().join("public");
  211. site.set_output_path(&public);
  212. site.build().unwrap();
  213. assert!(Path::new(&public).exists());
  214. assert_eq!(site.tags.unwrap().len(), 3);
  215. assert!(file_exists!(public, "index.html"));
  216. assert!(file_exists!(public, "sitemap.xml"));
  217. assert!(file_exists!(public, "robots.txt"));
  218. assert!(file_exists!(public, "a-fixed-url/index.html"));
  219. assert!(file_exists!(public, "posts/python/index.html"));
  220. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  221. assert!(file_exists!(public, "posts/with-assets/index.html"));
  222. // Sections
  223. assert!(file_exists!(public, "posts/index.html"));
  224. assert!(file_exists!(public, "posts/tutorials/index.html"));
  225. assert!(file_exists!(public, "posts/tutorials/devops/index.html"));
  226. assert!(file_exists!(public, "posts/tutorials/programming/index.html"));
  227. // TODO: add assertion for syntax highlighting
  228. // Tags are there
  229. assert!(file_exists!(public, "tags/index.html"));
  230. assert!(file_exists!(public, "tags/tag1/index.html"));
  231. assert!(file_exists!(public, "tags/tag2/index.html"));
  232. assert!(file_exists!(public, "tags/tag-with-space/index.html"));
  233. // Categories aren't
  234. assert_eq!(file_exists!(public, "categories/index.html"), false);
  235. // Tags are in the sitemap
  236. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/tags/</loc>"));
  237. assert!(file_contains!(public, "sitemap.xml", "<loc>https://replace-this-with-your-url.com/tags/tag-with-space/</loc>"));
  238. }
  239. #[test]
  240. fn can_build_site_and_insert_anchor_links() {
  241. let mut path = env::current_dir().unwrap().to_path_buf();
  242. path.push("test_site");
  243. let mut site = Site::new(&path, "config.toml").unwrap();
  244. site.load().unwrap();
  245. let tmp_dir = TempDir::new("example").expect("create temp dir");
  246. let public = &tmp_dir.path().join("public");
  247. site.set_output_path(&public);
  248. site.build().unwrap();
  249. assert!(Path::new(&public).exists());
  250. // anchor link inserted
  251. assert!(file_contains!(public, "posts/something-else/index.html", "<h1 id=\"title\"><a class=\"gutenberg-anchor\" href=\"#title\""));
  252. }
  253. #[test]
  254. fn can_build_site_with_pagination_for_section() {
  255. let mut path = env::current_dir().unwrap().to_path_buf();
  256. path.push("test_site");
  257. let mut site = Site::new(&path, "config.toml").unwrap();
  258. site.load().unwrap();
  259. for section in site.sections.values_mut(){
  260. if section.is_index() {
  261. continue;
  262. }
  263. section.meta.paginate_by = Some(2);
  264. section.meta.template = Some("section_paginated.html".to_string());
  265. }
  266. let tmp_dir = TempDir::new("example").expect("create temp dir");
  267. let public = &tmp_dir.path().join("public");
  268. site.set_output_path(&public);
  269. site.build().unwrap();
  270. assert!(Path::new(&public).exists());
  271. assert!(file_exists!(public, "index.html"));
  272. assert!(file_exists!(public, "sitemap.xml"));
  273. assert!(file_exists!(public, "robots.txt"));
  274. assert!(file_exists!(public, "a-fixed-url/index.html"));
  275. assert!(file_exists!(public, "posts/python/index.html"));
  276. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  277. assert!(file_exists!(public, "posts/with-assets/index.html"));
  278. // Sections
  279. assert!(file_exists!(public, "posts/index.html"));
  280. // And pagination!
  281. assert!(file_exists!(public, "posts/page/1/index.html"));
  282. // even if there is no pages, only the section!
  283. assert!(file_exists!(public, "paginated/page/1/index.html"));
  284. assert!(file_exists!(public, "paginated/index.html"));
  285. // should redirect to posts/
  286. assert!(file_contains!(
  287. public,
  288. "posts/page/1/index.html",
  289. "http-equiv=\"refresh\" content=\"0;url=https://replace-this-with-your-url.com/posts/\""
  290. ));
  291. assert!(file_contains!(public, "posts/index.html", "Num pagers: 3"));
  292. assert!(file_contains!(public, "posts/index.html", "Page size: 2"));
  293. assert!(file_contains!(public, "posts/index.html", "Current index: 1"));
  294. assert!(file_contains!(public, "posts/index.html", "has_next"));
  295. assert!(file_contains!(public, "posts/index.html", "First: https://replace-this-with-your-url.com/posts/"));
  296. assert!(file_contains!(public, "posts/index.html", "Last: https://replace-this-with-your-url.com/posts/page/3/"));
  297. assert_eq!(file_contains!(public, "posts/index.html", "has_prev"), false);
  298. assert!(file_exists!(public, "posts/page/2/index.html"));
  299. assert!(file_contains!(public, "posts/page/2/index.html", "Num pagers: 3"));
  300. assert!(file_contains!(public, "posts/page/2/index.html", "Page size: 2"));
  301. assert!(file_contains!(public, "posts/page/2/index.html", "Current index: 2"));
  302. assert!(file_contains!(public, "posts/page/2/index.html", "has_prev"));
  303. assert!(file_contains!(public, "posts/page/2/index.html", "has_next"));
  304. assert!(file_contains!(public, "posts/page/2/index.html", "First: https://replace-this-with-your-url.com/posts/"));
  305. assert!(file_contains!(public, "posts/page/2/index.html", "Last: https://replace-this-with-your-url.com/posts/page/3/"));
  306. }
  307. #[test]
  308. fn can_build_site_with_pagination_for_index() {
  309. let mut path = env::current_dir().unwrap().to_path_buf();
  310. path.push("test_site");
  311. let mut site = Site::new(&path, "config.toml").unwrap();
  312. site.load().unwrap();
  313. {
  314. let mut index = site.sections.get_mut(&path.join("content").join("_index.md")).unwrap();
  315. index.meta.paginate_by = Some(2);
  316. index.meta.template = Some("index_paginated.html".to_string());
  317. }
  318. let tmp_dir = TempDir::new("example").expect("create temp dir");
  319. let public = &tmp_dir.path().join("public");
  320. site.set_output_path(&public);
  321. site.build().unwrap();
  322. assert!(Path::new(&public).exists());
  323. assert!(file_exists!(public, "index.html"));
  324. assert!(file_exists!(public, "sitemap.xml"));
  325. assert!(file_exists!(public, "robots.txt"));
  326. assert!(file_exists!(public, "a-fixed-url/index.html"));
  327. assert!(file_exists!(public, "posts/python/index.html"));
  328. assert!(file_exists!(public, "posts/tutorials/devops/nix/index.html"));
  329. assert!(file_exists!(public, "posts/with-assets/index.html"));
  330. // And pagination!
  331. assert!(file_exists!(public, "page/1/index.html"));
  332. // even if there is no pages, only the section!
  333. assert!(file_exists!(public, "paginated/page/1/index.html"));
  334. assert!(file_exists!(public, "paginated/index.html"));
  335. // should redirect to index
  336. assert!(file_contains!(
  337. public,
  338. "page/1/index.html",
  339. "http-equiv=\"refresh\" content=\"0;url=https://replace-this-with-your-url.com/\""
  340. ));
  341. assert!(file_contains!(public, "index.html", "Num pages: 1"));
  342. assert!(file_contains!(public, "index.html", "Current index: 1"));
  343. assert!(file_contains!(public, "index.html", "First: https://replace-this-with-your-url.com/"));
  344. assert!(file_contains!(public, "index.html", "Last: https://replace-this-with-your-url.com/"));
  345. assert_eq!(file_contains!(public, "index.html", "has_prev"), false);
  346. assert_eq!(file_contains!(public, "index.html", "has_next"), false);
  347. }
  348. #[test]
  349. fn can_build_rss_feed() {
  350. let mut path = env::current_dir().unwrap().to_path_buf();
  351. path.push("test_site");
  352. let mut site = Site::new(&path, "config.toml").unwrap();
  353. site.load().unwrap();
  354. let tmp_dir = TempDir::new("example").expect("create temp dir");
  355. let public = &tmp_dir.path().join("public");
  356. site.set_output_path(&public);
  357. site.build().unwrap();
  358. assert!(Path::new(&public).exists());
  359. assert!(file_exists!(public, "rss.xml"));
  360. // latest article is posts/simple.md
  361. assert!(file_contains!(public, "rss.xml", "Simple article with shortcodes"));
  362. // Next is posts/python.md
  363. assert!(file_contains!(public, "rss.xml", "Python in posts"));
  364. }