This page is no longer maintained — Please continue to the home page at www.scala-lang.org

Emacs, Scala and Maven, Oh My!

2 replies
Grey
Joined: 2009-01-03,
User offline. Last seen 42 years 45 weeks ago.

Not a elisp guy by any stretch (first timer) ... but here is a rough, simple,
yet effective mvn.el file that I use with the Maven Scala plugin for my
Scala development. It runs maven from inside emacs. Its a script that I
found and extended a bit. You can click on errors and jump to the buffer,
M-x next-error etc. Simple coloring for scala-tests.

I load it in my ~/.emacs and bind it to key as follows.

(let ((path "/usr/local/share/emacs/site-lisp/mvn"))
(setq load-path (cons path load-path))
(load "mvn.el"))

(add-hook 'scala-mode-hook
'(lambda ()
(define-key scala-mode-map '[f6] 'mvn)))

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; MVN.EL Starts Here
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;; Really simple, yet effective Maven2 support.

(require 'compile)

(defvar mvn-command-history nil
"Maven command history variable")

(defvar scala-compile-error-regex
'("^\\[WARNING\\] \\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1 2
nil 2 nil))

(defun mvn-compile-buffer-name (mode)
"*Maven Compile*")

(set 'scala-compile-error-regex '("^\\[WARNING\\]
\\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1 2 nil 2 nil))

(set 'compilation-buffer-name-function 'mvn-compile-buffer-name)
(set 'compilation-error-regexp-alist (list scala-compile-error-regex))

(set 'compilation-mode-font-lock-keywords
'(("^\\[ERROR\\] BUILD FAILURE"
(0 compilation-error-face))
("^\\[WARNING\\]"
(0 compilation-warning-face))
("^\\(\\[INFO\\]\\)\\([^\n]*\\)"
(1 compilation-info-face)
(2 compilation-line-face))
("^\\[ERROR\\]"
(0 compilation-error-face))
("\\(Failures\\): [1-9][0-9]*,"
(0 compilation-error-face))
("\\(Failures\\): 0"
(0 compilation-info-face))
("\\(Errors\\): [1-9][0-9]*,"
(0 compilation-error-face))
("\\(Errors\\): 0"
(0 compilation-info-face))
("\\(Skipped\\): [1-9][0-9]*,"
(0 compilation-error-face))
("\\(Skipped\\): 0"
(0 compilation-info-face))
("Tests run: [0-9]+"
(0 compilation-warning-face))
("T E S T S"
(0 compilation-warning-face))
("Compilation finished at [^\n]+"
(0 compilation-info-face))
("^Compilation \\(exited
abnormally\\|interrupt\\|killed\\|terminated\\|segmentation
fault\\)\\(?:.*with code \\([0-9]+\\)\\)?.*"
(0 '(face nil message nil help-echo nil mouse-face nil) t)
(1 compilation-error-face)
(2 compilation-error-face nil t))))

(defun mvn-at-root (path)
"Determine if the given path is root."
(equal path (mvn-parent-path path)))

(defun mvn-parent-path (path)
"The parent path for the given path."
(file-truename (concat path "/..")))

(defun mvn-pom-at-path-p (path)
"Does a pom.xml exist in the given path."
(file-exists-p (concat path "/pom.xml")))

(defun mvn-find-path-to-pom ()
"Move up the directory tree for the current buffer until root or a pom.xml
is found."
(let ((fn (buffer-file-name)))
(let ((path (file-name-directory fn)))
(while (and (not (mvn-pom-at-path-p path))
(not (mvn-at-root path)))
(setf path (file-truename (mvn-parent-path path))))
path)))

(defun mvn-master (&optional args)
(interactive)
"Determine the potential master pom by checking if there is a pom in a
pom's parent directory. nil if no parent pom else the path to the parent."
(let ((pom-path (mvn-find-path-to-pom)))
(let ((parent-pom-path (file-truename (concat pom-path "/.."))))
(if (not (mvn-pom-at-path-p parent-pom-path))
(message "No master pom found.")
(compile (read-from-minibuffer "Command: "
(concat "mvn -f " parent-pom-path "/pom.xml clean install")
nil nil 'mvn-command-history))))))

(defun mvn (cmd)
"Runs maven in the current project. Starting at the directory where the
file being visited resides, a search is
made for pom.xml recursively. A maven command is made from the first
directory where the pom.xml file is found is then displayed
in the minibuffer. The command can be edited as needed and then executed.
Errors are navigate to as in any other compile mode"
(interactive "n0-clean 1-install 2-bundle:install 3-compile 4-test : ")
(let ((path (mvn-find-path-to-pom)))
(let ((goal (case cmd
((0) "clean")
((1) "install")
((2) "bundle:install")
((3) "compile")
((4) "test"))))
(if (not (file-exists-p (concat path "/pom.xml")))
(message "No pom.xml found")
(compile (read-from-minibuffer "Command: "
(concat "mvn -f " path (concat "/pom.xml " goal))
nil nil 'mvn-command-history))))))

;; String pattern for locating errors in maven output. This assumes a
Windows drive letter at the beginning
;;(add-to-list
;; 'compilation-error-regexp-alist
;; '("^\\([a-zA-Z]:.*\\):\\[\\([0-9]+\\),\\([0-9]+\\)\\]" 1 2 3))

Anders Bach Nielsen
Joined: 2008-12-17,
User offline. Last seen 42 years 45 weeks ago.
Re: Emacs, Scala and Maven, Oh My!

Hey

This look really interesting! I will look into this when I have a little
more time (read in a couple of days).

Just a question, because I dont know maven. This maven support for
emacs, is it generic og scala specific ? what are the scala specific
aspects of this ?

/Anders

Grey wrote:
>
> Not a elisp guy by any stretch (first timer) ... but here is a rough, simple,
> yet effective mvn.el file that I use with the Maven Scala plugin for my
> Scala development. It runs maven from inside emacs. Its a script that I
> found and extended a bit. You can click on errors and jump to the buffer,
> M-x next-error etc. Simple coloring for scala-tests.
>
> I load it in my ~/.emacs and bind it to key as follows.
>
> (let ((path "/usr/local/share/emacs/site-lisp/mvn"))
> (setq load-path (cons path load-path))
> (load "mvn.el"))
>
> (add-hook 'scala-mode-hook
> '(lambda ()
> (define-key scala-mode-map '[f6] 'mvn)))
>
>
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
> ;;; MVN.EL Starts Here
> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
>
> ;; Really simple, yet effective Maven2 support.
>
> (require 'compile)
>
> (defvar mvn-command-history nil
> "Maven command history variable")
>
> (defvar scala-compile-error-regex
> '("^\\[WARNING\\] \\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1 2
> nil 2 nil))
>
> (defun mvn-compile-buffer-name (mode)
> "*Maven Compile*")
>
> (set 'scala-compile-error-regex '("^\\[WARNING\\]
> \\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1 2 nil 2 nil))
>
> (set 'compilation-buffer-name-function 'mvn-compile-buffer-name)
> (set 'compilation-error-regexp-alist (list scala-compile-error-regex))
>
> (set 'compilation-mode-font-lock-keywords
> '(("^\\[ERROR\\] BUILD FAILURE"
> (0 compilation-error-face))
> ("^\\[WARNING\\]"
> (0 compilation-warning-face))
> ("^\\(\\[INFO\\]\\)\\([^\n]*\\)"
> (1 compilation-info-face)
> (2 compilation-line-face))
> ("^\\[ERROR\\]"
> (0 compilation-error-face))
> ("\\(Failures\\): [1-9][0-9]*,"
> (0 compilation-error-face))
> ("\\(Failures\\): 0"
> (0 compilation-info-face))
> ("\\(Errors\\): [1-9][0-9]*,"
> (0 compilation-error-face))
> ("\\(Errors\\): 0"
> (0 compilation-info-face))
> ("\\(Skipped\\): [1-9][0-9]*,"
> (0 compilation-error-face))
> ("\\(Skipped\\): 0"
> (0 compilation-info-face))
> ("Tests run: [0-9]+"
> (0 compilation-warning-face))
> ("T E S T S"
> (0 compilation-warning-face))
> ("Compilation finished at [^\n]+"
> (0 compilation-info-face))
> ("^Compilation \\(exited
> abnormally\\|interrupt\\|killed\\|terminated\\|segmentation
> fault\\)\\(?:.*with code \\([0-9]+\\)\\)?.*"
> (0 '(face nil message nil help-echo nil mouse-face nil) t)
> (1 compilation-error-face)
> (2 compilation-error-face nil t))))
>
> (defun mvn-at-root (path)
> "Determine if the given path is root."
> (equal path (mvn-parent-path path)))
>
> (defun mvn-parent-path (path)
> "The parent path for the given path."
> (file-truename (concat path "/..")))
>
> (defun mvn-pom-at-path-p (path)
> "Does a pom.xml exist in the given path."
> (file-exists-p (concat path "/pom.xml")))
>
> (defun mvn-find-path-to-pom ()
> "Move up the directory tree for the current buffer until root or a pom.xml
> is found."
> (let ((fn (buffer-file-name)))
> (let ((path (file-name-directory fn)))
> (while (and (not (mvn-pom-at-path-p path))
> (not (mvn-at-root path)))
> (setf path (file-truename (mvn-parent-path path))))
> path)))
>
> (defun mvn-master (&optional args)
> (interactive)
> "Determine the potential master pom by checking if there is a pom in a
> pom's parent directory. nil if no parent pom else the path to the parent."
> (let ((pom-path (mvn-find-path-to-pom)))
> (let ((parent-pom-path (file-truename (concat pom-path "/.."))))
> (if (not (mvn-pom-at-path-p parent-pom-path))
> (message "No master pom found.")
> (compile (read-from-minibuffer "Command: "
> (concat "mvn -f " parent-pom-path "/pom.xml clean install")
> nil nil 'mvn-command-history))))))
>
> (defun mvn (cmd)
> "Runs maven in the current project. Starting at the directory where the
> file being visited resides, a search is
> made for pom.xml recursively. A maven command is made from the first
> directory where the pom.xml file is found is then displayed
> in the minibuffer. The command can be edited as needed and then executed.
> Errors are navigate to as in any other compile mode"
> (interactive "n0-clean 1-install 2-bundle:install 3-compile 4-test : ")
> (let ((path (mvn-find-path-to-pom)))
> (let ((goal (case cmd
> ((0) "clean")
> ((1) "install")
> ((2) "bundle:install")
> ((3) "compile")
> ((4) "test"))))
> (if (not (file-exists-p (concat path "/pom.xml")))
> (message "No pom.xml found")
> (compile (read-from-minibuffer "Command: "
> (concat "mvn -f " path (concat "/pom.xml " goal))
> nil nil 'mvn-command-history))))))
>
> ;; String pattern for locating errors in maven output. This assumes a
> Windows drive letter at the beginning
> ;;(add-to-list
> ;; 'compilation-error-regexp-alist
> ;; '("^\\([a-zA-Z]:.*\\):\\[\\([0-9]+\\),\\([0-9]+\\)\\]" 1 2 3))
>
>
>
>
>

Grey
Joined: 2009-01-03,
User offline. Last seen 42 years 45 weeks ago.
Re: Emacs, Scala and Maven, Oh My!

Finding and running the pom.xml is generic. Being able to click on compile
errors, moving through errors is generic using the standard emacs
compile.el, requiring only the appropriate regexp to identify Scala compiler
error msgs.

There is additional window dressing provided by font-lock regexps for the
Scala compiler / ScalaTest output generally following the convention, Green
good, Red bad.

Anders Bach Nielsen-2 wrote:
>
> Hey
>
> This look really interesting! I will look into this when I have a little
> more time (read in a couple of days).
>
> Just a question, because I dont know maven. This maven support for
> emacs, is it generic og scala specific ? what are the scala specific
> aspects of this ?
>
> /Anders
>
> Grey wrote:
>>
>> Not a elisp guy by any stretch (first timer) ... but here is a rough,
>> simple,
>> yet effective mvn.el file that I use with the Maven Scala plugin for my
>> Scala development. It runs maven from inside emacs. Its a script that I
>> found and extended a bit. You can click on errors and jump to the
>> buffer,
>> M-x next-error etc. Simple coloring for scala-tests.
>>
>> I load it in my ~/.emacs and bind it to key as follows.
>>
>> (let ((path "/usr/local/share/emacs/site-lisp/mvn"))
>> (setq load-path (cons path load-path))
>> (load "mvn.el"))
>>
>> (add-hook 'scala-mode-hook
>> '(lambda ()
>> (define-key scala-mode-map '[f6] 'mvn)))
>>
>>
>> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
>> ;;; MVN.EL Starts Here
>> ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
>>
>> ;; Really simple, yet effective Maven2 support.
>>
>> (require 'compile)
>>
>> (defvar mvn-command-history nil
>> "Maven command history variable")
>>
>> (defvar scala-compile-error-regex
>> '("^\\[WARNING\\] \\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1
>> 2
>> nil 2 nil))
>>
>> (defun mvn-compile-buffer-name (mode)
>> "*Maven Compile*")
>>
>> (set 'scala-compile-error-regex '("^\\[WARNING\\]
>> \\([a-zA-Z0-9/-]+[.]scala\\):\\([0-9]+\\): error:" 1 2 nil 2 nil))
>>
>> (set 'compilation-buffer-name-function 'mvn-compile-buffer-name)
>> (set 'compilation-error-regexp-alist (list scala-compile-error-regex))
>>
>> (set 'compilation-mode-font-lock-keywords
>> '(("^\\[ERROR\\] BUILD FAILURE"
>> (0 compilation-error-face))
>> ("^\\[WARNING\\]"
>> (0 compilation-warning-face))
>> ("^\\(\\[INFO\\]\\)\\([^\n]*\\)"
>> (1 compilation-info-face)
>> (2 compilation-line-face))
>> ("^\\[ERROR\\]"
>> (0 compilation-error-face))
>> ("\\(Failures\\): [1-9][0-9]*,"
>> (0 compilation-error-face))
>> ("\\(Failures\\): 0"
>> (0 compilation-info-face))
>> ("\\(Errors\\): [1-9][0-9]*,"
>> (0 compilation-error-face))
>> ("\\(Errors\\): 0"
>> (0 compilation-info-face))
>> ("\\(Skipped\\): [1-9][0-9]*,"
>> (0 compilation-error-face))
>> ("\\(Skipped\\): 0"
>> (0 compilation-info-face))
>> ("Tests run: [0-9]+"
>> (0 compilation-warning-face))
>> ("T E S T S"
>> (0 compilation-warning-face))
>> ("Compilation finished at [^\n]+"
>> (0 compilation-info-face))
>> ("^Compilation \\(exited
>> abnormally\\|interrupt\\|killed\\|terminated\\|segmentation
>> fault\\)\\(?:.*with code \\([0-9]+\\)\\)?.*"
>> (0 '(face nil message nil help-echo nil mouse-face nil) t)
>> (1 compilation-error-face)
>> (2 compilation-error-face nil t))))
>>
>> (defun mvn-at-root (path)
>> "Determine if the given path is root."
>> (equal path (mvn-parent-path path)))
>>
>> (defun mvn-parent-path (path)
>> "The parent path for the given path."
>> (file-truename (concat path "/..")))
>>
>> (defun mvn-pom-at-path-p (path)
>> "Does a pom.xml exist in the given path."
>> (file-exists-p (concat path "/pom.xml")))
>>
>> (defun mvn-find-path-to-pom ()
>> "Move up the directory tree for the current buffer until root or a
>> pom.xml
>> is found."
>> (let ((fn (buffer-file-name)))
>> (let ((path (file-name-directory fn)))
>> (while (and (not (mvn-pom-at-path-p path))
>> (not (mvn-at-root path)))
>> (setf path (file-truename (mvn-parent-path path))))
>> path)))
>>
>> (defun mvn-master (&optional args)
>> (interactive)
>> "Determine the potential master pom by checking if there is a pom in a
>> pom's parent directory. nil if no parent pom else the path to the
>> parent."
>> (let ((pom-path (mvn-find-path-to-pom)))
>> (let ((parent-pom-path (file-truename (concat pom-path "/.."))))
>> (if (not (mvn-pom-at-path-p parent-pom-path))
>> (message "No master pom found.")
>> (compile (read-from-minibuffer "Command: "
>> (concat "mvn -f " parent-pom-path "/pom.xml clean install")
>> nil nil 'mvn-command-history))))))
>>
>> (defun mvn (cmd)
>> "Runs maven in the current project. Starting at the directory where the
>> file being visited resides, a search is
>> made for pom.xml recursively. A maven command is made from the first
>> directory where the pom.xml file is found is then displayed
>> in the minibuffer. The command can be edited as needed and then
>> executed.
>> Errors are navigate to as in any other compile mode"
>> (interactive "n0-clean 1-install 2-bundle:install 3-compile 4-test : ")
>> (let ((path (mvn-find-path-to-pom)))
>> (let ((goal (case cmd
>> ((0) "clean")
>> ((1) "install")
>> ((2) "bundle:install")
>> ((3) "compile")
>> ((4) "test"))))
>> (if (not (file-exists-p (concat path "/pom.xml")))
>> (message "No pom.xml found")
>> (compile (read-from-minibuffer "Command: "
>> (concat "mvn -f " path (concat "/pom.xml " goal))
>> nil nil 'mvn-command-history))))))
>>
>> ;; String pattern for locating errors in maven output. This assumes a
>> Windows drive letter at the beginning
>> ;;(add-to-list
>> ;; 'compilation-error-regexp-alist
>> ;; '("^\\([a-zA-Z]:.*\\):\\[\\([0-9]+\\),\\([0-9]+\\)\\]" 1 2 3))
>>
>>
>>
>>
>>

Copyright © 2012 École Polytechnique Fédérale de Lausanne (EPFL), Lausanne, Switzerland