2007-01-01 23:48:42 -07:00
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defvar coverage-annotation-file ".coverage.el")
|
|
|
|
(defvar coverage-annotations nil)
|
2007-01-01 23:48:42 -07:00
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun find-coverage-annotation-file ()
|
2007-01-04 01:25:36 -07:00
|
|
|
(let ((dir (file-name-directory buffer-file-name))
|
|
|
|
(olddir "/"))
|
|
|
|
(while (and (not (equal dir olddir))
|
2010-01-28 09:39:04 -08:00
|
|
|
(not (file-regular-p (concat dir coverage-annotation-file))))
|
2007-01-04 01:25:36 -07:00
|
|
|
(setq olddir dir
|
|
|
|
dir (file-name-directory (directory-file-name dir))))
|
2010-01-28 09:39:04 -08:00
|
|
|
(and (not (equal dir olddir)) (concat dir coverage-annotation-file))
|
2007-01-04 01:25:36 -07:00
|
|
|
))
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun load-coverage-annotations ()
|
|
|
|
(let* ((annotation-file (find-coverage-annotation-file))
|
2007-01-04 01:25:36 -07:00
|
|
|
(coverage
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert-file-contents annotation-file)
|
|
|
|
(let ((form (read (current-buffer))))
|
|
|
|
(eval form)))))
|
2010-01-28 09:39:04 -08:00
|
|
|
(setq coverage-annotations coverage)
|
2007-01-01 23:48:42 -07:00
|
|
|
coverage
|
|
|
|
))
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun coverage-unannotate ()
|
2007-01-01 23:48:42 -07:00
|
|
|
(save-excursion
|
|
|
|
(dolist (ov (overlays-in (point-min) (point-max)))
|
|
|
|
(delete-overlay ov))
|
2010-01-28 09:39:04 -08:00
|
|
|
(setq coverage-this-buffer-is-annotated nil)
|
2007-01-04 21:52:40 -07:00
|
|
|
(message "Removed annotations")
|
2007-01-01 23:48:42 -07:00
|
|
|
))
|
|
|
|
|
2007-01-07 13:10:15 -07:00
|
|
|
;; in emacs22, it will be possible to put the annotations in the fringe. Set
|
|
|
|
;; a display property for one of the characters in the line, using
|
|
|
|
;; (right-fringe BITMAP FACE), where BITMAP should probably be right-triangle
|
|
|
|
;; or so, and FACE should probably be '(:foreground "red"). We can also
|
|
|
|
;; create new bitmaps, with faces. To do tartans will require a lot of
|
|
|
|
;; bitmaps, and you've only got about 8 pixels to work with.
|
|
|
|
|
|
|
|
;; unfortunately emacs21 gives us less control over the fringe. We can use
|
|
|
|
;; overlays to put letters on the left or right margins (in the text area,
|
|
|
|
;; overriding actual program text), and to modify the text being displayed
|
|
|
|
;; (by changing its background color, or adding a box around each word).
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun coverage-annotate (show-code)
|
|
|
|
(let ((allcoverage (load-coverage-annotations))
|
2010-02-23 23:47:57 -05:00
|
|
|
(filename-key (expand-file-name buffer-file-truename))
|
2007-01-04 01:25:36 -07:00
|
|
|
thiscoverage code-lines covered-lines uncovered-code-lines
|
|
|
|
)
|
|
|
|
(while (and (not (gethash filename-key allcoverage nil))
|
|
|
|
(string-match "/" filename-key))
|
|
|
|
;; eat everything up to and including the first slash, then look again
|
|
|
|
(setq filename-key (substring filename-key
|
|
|
|
(+ 1 (string-match "/" filename-key)))))
|
|
|
|
(setq thiscoverage (gethash filename-key allcoverage nil))
|
|
|
|
(if thiscoverage
|
|
|
|
(progn
|
2010-01-28 09:39:04 -08:00
|
|
|
(setq coverage-this-buffer-is-annotated t)
|
2007-01-04 01:25:36 -07:00
|
|
|
(setq code-lines (nth 0 thiscoverage)
|
|
|
|
covered-lines (nth 1 thiscoverage)
|
|
|
|
uncovered-code-lines (nth 2 thiscoverage)
|
|
|
|
)
|
|
|
|
|
|
|
|
(save-excursion
|
|
|
|
(dolist (ov (overlays-in (point-min) (point-max)))
|
|
|
|
(delete-overlay ov))
|
|
|
|
(if show-code
|
|
|
|
(dolist (line code-lines)
|
|
|
|
(goto-line line)
|
|
|
|
;;(add-text-properties (point) (line-end-position) '(face bold) )
|
|
|
|
(overlay-put (make-overlay (point) (line-end-position))
|
2007-01-01 23:48:42 -07:00
|
|
|
;'before-string "C"
|
|
|
|
;'face '(background-color . "green")
|
2007-01-04 01:25:36 -07:00
|
|
|
'face '(:background "dark green")
|
|
|
|
)
|
|
|
|
))
|
|
|
|
(dolist (line uncovered-code-lines)
|
|
|
|
(goto-line line)
|
|
|
|
(overlay-put (make-overlay (point) (line-end-position))
|
2007-01-01 23:48:42 -07:00
|
|
|
;'before-string "D"
|
2007-01-04 01:25:36 -07:00
|
|
|
;'face '(:background "blue")
|
|
|
|
;'face '(:underline "blue")
|
|
|
|
'face '(:box "red")
|
|
|
|
)
|
|
|
|
)
|
2010-08-04 00:11:31 -07:00
|
|
|
(message (format "Added annotations: %d uncovered lines"
|
|
|
|
(safe-length uncovered-code-lines)))
|
2007-01-04 01:25:36 -07:00
|
|
|
)
|
|
|
|
)
|
|
|
|
(message "unable to find coverage for this file"))
|
|
|
|
))
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun coverage-toggle-annotations (show-code)
|
2007-01-04 01:25:36 -07:00
|
|
|
(interactive "P")
|
2010-01-28 09:39:04 -08:00
|
|
|
(if coverage-this-buffer-is-annotated
|
|
|
|
(coverage-unannotate)
|
|
|
|
(coverage-annotate show-code))
|
2007-01-04 01:25:36 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(setq coverage-this-buffer-is-annotated nil)
|
|
|
|
(make-variable-buffer-local 'coverage-this-buffer-is-annotated)
|
2007-01-04 01:25:36 -07:00
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(define-minor-mode coverage-annotation-minor-mode
|
2007-01-04 01:25:36 -07:00
|
|
|
"Minor mode to annotate code-coverage information"
|
|
|
|
nil
|
2010-01-28 09:39:04 -08:00
|
|
|
" CA"
|
2007-01-04 01:25:36 -07:00
|
|
|
'(
|
2010-01-28 09:39:04 -08:00
|
|
|
("\C-c\C-a" . coverage-toggle-annotations)
|
2007-01-04 01:25:36 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
() ; forms run on mode entry/exit
|
|
|
|
)
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(defun maybe-enable-coverage-mode ()
|
2007-01-18 01:00:11 -07:00
|
|
|
(if (string-match "/src/allmydata/" (buffer-file-name))
|
2010-01-28 09:39:04 -08:00
|
|
|
(coverage-annotation-minor-mode t)
|
2007-01-18 01:00:11 -07:00
|
|
|
))
|
|
|
|
|
2010-01-28 09:39:04 -08:00
|
|
|
(add-hook 'python-mode-hook 'maybe-enable-coverage-mode)
|