;;; snakemake-mode.el --- Major mode for editing Snakemake files -*- lexical-binding: t; -*- ;; Copyright all Snakemake mode contributors ;; Author: Kyle Meyer ;; URL: https://git.kyleam.com/snakemake-mode/about ;; Keywords: tools ;; Version: 2.0.0 ;; Package-Requires: ((emacs "26.1") (transient "0.3.0")) ;; This program is free software; you can redistribute it and/or modify ;; it under the terms of the GNU General Public License as published by ;; the Free Software Foundation; either version 3, or (at your option) ;; any later version. ;; ;; This program is distributed in the hope that it will be useful, ;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;; GNU General Public License for more details. ;; ;; You should have received a copy of the GNU General Public License ;; along with GNU Emacs. If not, see . ;;; Commentary: ;; Snakemake mode provides support for editing Snakemake [1] files. It ;; builds on Python mode to provide fontification, indentation, and ;; imenu indexing for Snakemake's rule blocks. ;; ;; See also snakemake.el, which is packaged with snakemake-mode.el and ;; provides an interface for running Snakemake commands. ;; ;; If Snakemake mode is installed from MELPA, no additional setup is ;; required. It will be loaded the first time a file named 'Snakefile' ;; is opened. ;; ;; Otherwise, put snakemake-mode.el in your `load-path' and add ;; ;; (require 'snakemake-mode) ;; ;; to your initialization file. ;; ;; snakemake-mode.el also includes support for highlighting embedded R ;; code. See the snakemake-mode-setup-mmm function documentation for how. ;; ;; [1] https://snakemake.github.io/ ;;; Code: (require 'cl-lib) (require 'python) ;;; Customization ;;;###autoload (defgroup snakemake-mode nil "Support for Snakemake files" :group 'tools) (defcustom snakemake-mode-hook nil "Hook run when entering `snakemake-mode'." :type 'hook) (defcustom snakemake-indent-field-offset 4 "Offset for field indentation." :type 'integer) (defcustom snakemake-indent-value-offset 4 "Offset for field values that the line below the field key." :type 'integer) ;;; Regexp (eval-and-compile (defconst snakemake-rx-constituents `((named-rule . ,(rx (and (group symbol-start (or "checkpoint" "module" "rule" "subworkflow")) " " (group (one-or-more (or (syntax word) (syntax symbol))))))) (anon-rule . ,(rx symbol-start "rule")) (field-key . ,(rx symbol-start (or "benchmark" "cache" "conda" "container" "cwl" "default_target" "envmodules" "group" "handover" "input" "localrule" "log" "message" "name" "notebook" "output" "params" "priority" "resources" "run" "script" "shadow" "shell" "singularity" "threads" "version" "wildcard_constraints" "wrapper" ;; Keys for subworkflow blocks "configfile" "snakefile" "workdir") symbol-end)) (sm-command . ,(rx symbol-start (or "configfile" "container" "containerized" "envvars" "include" "localrules" "onerror" "onsuccess" "report" "ruleorder" "singularity" "wildcard_constraints" "workdir") symbol-end)) (sm-builtin . ,(rx symbol-start (or "ancient" "checkpoints" "directory" "dynamic" "expand" "input" "multiext" "output" "params" "pipe" "protected" "report" "shell" "temp" "touch" "unpack" "wildcards") symbol-end))) "Snakemake-specific sexps for `snakemake-rx'.") (defmacro snakemake-rx (&rest regexps) "Specialized `rx' for Snakemake mode." ;; Modified from `python-rx'. (let ((rx-constituents (append snakemake-rx-constituents rx-constituents))) (cond ((null regexps) (error "No regexp")) ((cdr regexps) (rx-to-string `(and ,@regexps) t)) (t (rx-to-string (car regexps) t)))))) (defconst snakemake-rule-or-subworkflow-re (snakemake-rx line-start (zero-or-more space) (or named-rule (group anon-rule)) (zero-or-more space) ":") "Regexp matching a rule or subworkflow.") ;;; Info and navigation (defun snakemake-block-info () "Return description of rule or subworkflow block at point." (save-excursion (save-restriction (widen) (let ((pos (point))) (end-of-line) (and (re-search-backward snakemake-rule-or-subworkflow-re nil t) (let ((type (or (match-string-no-properties 1) "rule")) (name (match-string-no-properties 2)) (start (or (match-beginning 1) (match-beginning 3))) (rule-indent (current-indentation)) end) (beginning-of-line) (forward-line) (while (and (or (< rule-indent (current-indentation)) (looking-at-p "^\\s-*$")) (or (not (eobp)) (progn (setq end (point-max)) nil))) (setq end (line-end-position)) (forward-line)) (when (<= start pos end) (list type name start end)))))))) (defun snakemake-beginning-of-block (&optional arg) "Move to beginning of rule block. With ARG, do it that many times. Negative ARG signals to move forward rather than backward." (when (or (null arg) (zerop arg)) (setq arg 1)) (funcall (if (> arg 0) #'re-search-backward (lambda (&rest args) (end-of-line) (prog1 (apply #'re-search-forward args) (beginning-of-line)))) snakemake-rule-or-subworkflow-re nil 'move (abs arg)) (looking-at-p snakemake-rule-or-subworkflow-re)) (defun snakemake-end-of-block () "Move to end of rule or subworkflow block." (let ((end (nth 3 (snakemake-block-info)))) (when end (goto-char end)))) (defun snakemake-beginning-of-defun (&optional arg) "Move to beginning of current rule block or function. With ARG, do it that many times. Negative ARG signals to move forward rather than backward." (when (or (null arg) (zerop arg)) (setq arg 1)) (let ((choose-fn (if (> arg 0) #'max #'min)) (cands (delq nil (mapcar (lambda (f) (save-excursion (and (funcall f arg) (point)))) (list #'snakemake-beginning-of-block #'python-nav-beginning-of-defun))))) (cond (cands (goto-char (apply choose-fn cands))) ((> arg 0) (goto-char (point-min)) nil) (t (goto-char (point-max)) nil)))) (defun snakemake-end-of-defun () "Move to end of current rule block or function." (or (snakemake-end-of-block) (python-nav-end-of-defun))) (defun snakemake-block-or-defun-name () "Return name of current rule or function. This function is appropriate to use as the value of `add-log-current-defun-function'." (or (nth 1 (snakemake-block-info)) (python-info-current-defun))) ;;; Indentation (defun snakemake--calculate-indentation (&optional previous) "Return indentation offset for the current line. A non-nil value for PREVIOUS indicates that the previous command was an indentation command. When Python mode should handle the indentation, a nil value is returned." (when (memq (car (python-indent-context)) (list :after-line ;; If point is on a value line following a naked ;; field value, `python-indent-context' returns ;; :after-block-start. :after-block-start)) (let* ((initial-indent (current-indentation)) (goto-first-p (or (not previous) (zerop initial-indent)))) (save-excursion (save-restriction (widen) (beginning-of-line) (if (looking-at-p (snakemake-rx line-start (zero-or-more space) (or (and field-key (zero-or-more space) ":") (or "\"\"\"" "'''")))) (and goto-first-p (let (rule-indent) (while (not (or rule-indent (bobp))) (forward-line -1) (when (looking-at-p snakemake-rule-or-subworkflow-re) (setq rule-indent (current-indentation)))) (and rule-indent (+ rule-indent snakemake-indent-field-offset)))) ;; We need to look back to determine indentation. (skip-chars-backward " \t\n") (beginning-of-line) (cond ((looking-at-p (snakemake-rx line-start (zero-or-more space) (or named-rule anon-rule field-key sm-command) (zero-or-more space) ":" (zero-or-more space) (zero-or-one (and "#" (zero-or-more not-newline))) line-end)) (let ((above-indent (current-indentation))) (cond (goto-first-p (+ above-indent snakemake-indent-value-offset)) ((< above-indent initial-indent) above-indent)))) ((looking-at (snakemake-rx line-start (zero-or-more space) field-key (zero-or-more space) ":" (zero-or-more space))) (let ((above-indent (current-indentation))) (cond (goto-first-p (- (match-end 0) (line-beginning-position))) ((< above-indent initial-indent) above-indent)))) ((save-excursion (let ((above-indent (current-indentation)) field-indent) (when (> above-indent 0) (while (and (not (bobp)) (or (= above-indent (setq field-indent (current-indentation))) (looking-at-p "^\\s-*$"))) (forward-line -1))) (and (looking-at (snakemake-rx line-start (zero-or-more space) (group field-key) (zero-or-more space) ":" (zero-or-more space) (group (zero-or-more not-newline)))) (not (equal (match-string-no-properties 1) "run")) (cond (goto-first-p (let ((field-val (match-string-no-properties 2))) (if (or (equal field-val "") (string-match-p "\\`#" field-val)) above-indent (- (match-beginning 2) (line-beginning-position))))) ((< field-indent initial-indent) field-indent))))))))))))) (defun snakemake-indent-line (&optional previous) "Snakemake mode variant of `python-indent-line'." (let ((follow-indentation-p (and (<= (line-beginning-position) (point)) (>= (+ (line-beginning-position) (current-indentation)) (point))))) (save-excursion (indent-line-to (or (snakemake--calculate-indentation previous) (python-indent-calculate-indentation previous)))) (when follow-indentation-p (back-to-indentation)))) (defun snakemake-indent-line-function () "Snakemake mode variant of `python-indent-line-function'." (snakemake-indent-line (and (memq this-command python-indent-trigger-commands) (eq last-command this-command)))) ;;; Imenu (defun snakemake-imenu-create-index () "Create Imenu index for rule blocks. If `python-imenu-create-index' returns a non-nil value, also include these results and append a \"(rule)\" to the index label." (let ((py-index (python-imenu-create-index)) (sm-index (snakemake--imenu-build-rule-index))) (if py-index (append (mapcar (lambda (x) (cons (concat (car x) " (rule)") (cdr x))) sm-index) py-index) sm-index))) (defun snakemake--imenu-build-rule-index () (goto-char (point-min)) (let (index) (while (re-search-forward snakemake-rule-or-subworkflow-re nil t) (push (cons (match-string-no-properties 2) (save-excursion (beginning-of-line) (point-marker))) index)) (nreverse index))) ;;; Embedded language syntax-highlighting (declare-function mmm-add-classes "mmm-vars") (declare-function mmm-add-mode-ext-class "mmm-vars") (defun snakemake-mode-setup-mmm () "Set up MMM mode to highlight embedded R code. You must have the R-strings either within a R(\"\"\" \"\"\") function call or a code block delimited with \"\"\"#r and \"\"\". (Triple single-quotes also accepted.) For automatic highlighting of embedded regions, you need to set `mmm-global-mode' to `maybe'." (unless (require 'mmm-mode nil t) (user-error "You need to install mmm-mode")) (unless (bound-and-true-p mmm-global-mode) (display-warning 'snakemake-mode "To get automatic syntax highlighting of embedded R, you need to set mmm-global-mode to a non-nil value such as `maybe'.")) (mmm-add-classes '((snakemake-R-call-double :submode R-mode :front ".*R\(\"\"\"" :back ".*\"\"\"\)"))) (mmm-add-classes '((snakemake-R-call-regular :submode R-mode :front ".*R\('''" :back ".*'''\)"))) (mmm-add-classes '((snakemake-R-string-double :submode R-mode :front ".*\"\"\" *# *[rR]" :back ".*\"\"\""))) (mmm-add-classes '((snakemake-R-string-regular :submode R-mode :front ".*''' *# *[rR]" :back ".*'''"))) (mmm-add-mode-ext-class 'snakemake-mode nil 'snakemake-R-call-double) (mmm-add-mode-ext-class 'snakemake-mode nil 'snakemake-R-call-regular) (mmm-add-mode-ext-class 'snakemake-mode nil 'snakemake-R-string-double) (mmm-add-mode-ext-class 'snakemake-mode nil 'snakemake-R-string-regular)) ;;; Mode (defvar snakemake--font-lock-keywords `((,snakemake-rule-or-subworkflow-re (1 font-lock-keyword-face nil 'lax) (2 font-lock-function-name-face nil 'lax) (3 font-lock-keyword-face nil 'lax)) (,(snakemake-rx line-start (one-or-more space) (group field-key) (zero-or-more space) ":") 1 font-lock-type-face) (,(snakemake-rx line-start (zero-or-more space) (group sm-command) (zero-or-more space) ":") 1 font-lock-keyword-face) (,(rx line-start (group "use rule ")) 1 font-lock-keyword-face) (,(snakemake-rx (group sm-builtin)) 1 font-lock-builtin-face))) (defvar snakemake-font-lock-keywords) (if (bound-and-true-p python-font-lock-keywords-level-1) (with-no-warnings ;; In Emacs 27 `python-font-lock-keywords' was split up into ;; different decoration levels. (defvar snakemake-font-lock-keywords-level-1 (append snakemake--font-lock-keywords python-font-lock-keywords-level-1)) (defvar snakemake-font-lock-keywords-level-2 (append snakemake--font-lock-keywords python-font-lock-keywords-level-2)) (defvar snakemake-font-lock-keywords-maximum-decoration (append snakemake--font-lock-keywords python-font-lock-keywords-maximum-decoration)) (setq snakemake-font-lock-keywords ;; Mirrors `python-font-lock-keywords'. '(snakemake-font-lock-keywords-level-1 snakemake-font-lock-keywords-level-1 snakemake-font-lock-keywords-level-2 snakemake-font-lock-keywords-maximum-decoration))) (setq snakemake-font-lock-keywords (append snakemake--font-lock-keywords python-font-lock-keywords))) ;;;###autoload (define-derived-mode snakemake-mode python-mode "Snakemake" "Mode for editing Snakemake files." (set (make-local-variable 'imenu-create-index-function) #'snakemake-imenu-create-index) (set (make-local-variable 'indent-line-function) 'snakemake-indent-line-function) (set (make-local-variable 'indent-region-function) nil) (set (make-local-variable 'beginning-of-defun-function) #'snakemake-beginning-of-defun) (set (make-local-variable 'end-of-defun-function) #'snakemake-end-of-defun) (set (make-local-variable 'add-log-current-defun-function) #'snakemake-block-or-defun-name) (set (make-local-variable 'font-lock-defaults) (cons snakemake-font-lock-keywords (cdr font-lock-defaults)))) ;;;###autoload (add-to-list 'auto-mode-alist '("Snakefile\\'" . snakemake-mode)) ;;;###autoload (add-to-list 'auto-mode-alist '("\\.\\(?:sm\\)?rules\\'" . snakemake-mode)) ;;;###autoload (add-to-list 'auto-mode-alist '("\\.smk\\'" . snakemake-mode)) ;;;###autoload (add-to-list 'auto-mode-alist '("\\.snakefile\\'" . snakemake-mode)) (provide 'snakemake-mode) ;;; snakemake-mode.el ends here