summaryrefslogtreecommitdiff
path: root/lisp/progmodes/sql.el
diff options
context:
space:
mode:
Diffstat (limited to 'lisp/progmodes/sql.el')
-rw-r--r--lisp/progmodes/sql.el160
1 files changed, 135 insertions, 25 deletions
diff --git a/lisp/progmodes/sql.el b/lisp/progmodes/sql.el
index 3a0cb87e861..c8b7a2e62eb 100644
--- a/lisp/progmodes/sql.el
+++ b/lisp/progmodes/sql.el
@@ -4,7 +4,7 @@
;; Author: Alex Schroeder <a.schroeder@bsiag.ch>
;; Maintainer: Alex Schroeder <a.schroeder@bsiag.ch>
-;; Version: 1.3.2
+;; Version: 1.4.0
;; Keywords: comm languages processes
;; This file is part of GNU Emacs.
@@ -152,6 +152,31 @@ the window is split and the SQLi buffer is shown. If this
variable is not nil, that buffer's window will be selected
by calling `pop-to-buffer'. If this variable is nil, that
buffer is shown using `display-buffer'."
+ :type 'boolean
+ :group 'SQL)
+
+(defcustom sql-input-ring-file-name nil
+ "*If non-nil, name of the file to read/write input history.
+
+This is used to locally set `comint-input-ring-file-name' when reading
+or writing the input history."
+ :type '(choice (const :tag "none" nil)
+ (file))
+ :group 'SQL)
+
+(defcustom sql-input-ring-separator "\n--\n"
+ "*Separator between commands in the history file.
+
+If set to \"\\n\", each line in the history file will be interpreted as
+one command. Multi-line commands are split into several commands when
+the input ring is initialized from a history file.
+
+This variable used to locally set `comint-input-ring-separator' when
+reading or writing the history file. `comint-input-ring-separator' is
+new in Emacs 20.4; if your Emacs does not have it, setting
+`sql-input-ring-separator' will have no effect. In that case multiline
+commands will be split into several commands when the input history is
+read, as if you had set `sql-input-ring-separator' to \"\\n\"."
:type 'string
:group 'SQL)
@@ -167,6 +192,14 @@ buffer is shown using `display-buffer'."
:type 'hook
:group 'SQL)
+(defcustom sql-set-sqli-hook '()
+ "*Hook for reacting to changes of `sql-buffer'.
+
+This is called by `sql-set-sqli-buffer' when the value of `sql-buffer'
+is changed."
+ :type 'hook
+ :group 'SQL)
+
;; Customisation for Oracle
(defcustom sql-oracle-program "sqlplus"
@@ -281,7 +314,7 @@ The global value of sql-buffer is the name of the latest SQLi buffer
created. Any SQL buffer created will make a local copy of this value.
See `sql-interactive-mode' for more on multiple sessions. If you want
to change the SQLi buffer a SQL mode sends its SQL strings to, change
-the local value of sql-buffer using \\[sql-change-sqli-buffer].")
+the local value of `sql-buffer' using \\[sql-set-sqli-buffer].")
(defvar sql-prompt-regexp nil
"Prompt used to initialize `comint-prompt-regexp'.
@@ -293,6 +326,11 @@ You can change `comint-prompt-regexp' on `sql-interactive-mode-hook'.")
You can change it on `sql-interactive-mode-hook'.")
+(defvar sql-alternate-buffer-name nil
+ "Buffer-local string used to possibly rename the SQLi buffer.
+
+Used by `sql-rename-buffer'.")
+
;; Keymap for sql-interactive-mode, based on comint-mode-map.
(if (not (string-match "XEmacs\\|Lucid" emacs-version))
@@ -337,12 +375,20 @@ You can change it on `sql-interactive-mode-hook'.")
["Send Buffer" sql-send-buffer (and (buffer-live-p sql-buffer)
(get-buffer-process sql-buffer))]
["Show SQLi buffer" sql-show-sqli-buffer t]
- ["Change SQLi buffer" sql-change-sqli-buffer t]
+ ["Set SQLi buffer" sql-set-sqli-buffer t]
["Pop to SQLi buffer after send"
sql-toggle-pop-to-buffer-after-send-region
:style toggle
:selected sql-pop-to-buffer-after-send-region]))
+;; easy menu for sql-interactive-mode.
+
+(easy-menu-define
+ sql-interactive-mode-menu sql-interactive-mode-map
+ "Menu for `sql-interactive-mode'."
+ '("SQL"
+ ["Rename Buffer" sql-rename-buffer t]))
+
;; Abbreviations -- if you want more of them, define them in your
;; ~/.emacs file. Abbrevs have to be enabled in your ~/.emacs, too.
@@ -647,22 +693,24 @@ function like this: (sql-get-login 'user 'password 'database)."
(setq sql-database
(read-from-minibuffer "Database: " sql-database nil nil
sql-database-history))))
+(defun sql-set-sqli-buffer ()
+ "Set the SQLi buffer SQL strings are sent to.
-(defun sql-change-sqli-buffer ()
- "Change the SQLi buffer SQL strings are sent to.
+Call this function in a SQL buffer in order to set the SQLi buffer SQL
+strings are sent to. Calling this function sets `sql-buffer' and runs
+`sql-set-sqli-hook'.
-Call this function in a SQL buffer in order to change the SQLi buffer
-SQL strings are sent to. Calling this function sets `sql-buffer'.
-
-If you call it from a SQL buffer, this changes the local copy of
+If you call it from a SQL buffer, this sets the local copy of
`sql-buffer'.
-If you call it from anywhere else, it changes the global copy of
+If you call it from anywhere else, it sets the global copy of
`sql-buffer'."
(interactive)
(let ((new-buffer (get-buffer (read-buffer "New SQLi buffer: " nil t))))
(if new-buffer
- (setq sql-buffer new-buffer))))
+ (progn
+ (setq sql-buffer new-buffer)
+ (run-hooks sql-set-sqli-hook)))))
(defun sql-show-sqli-buffer ()
"Show the name of current SQLi buffer.
@@ -671,11 +719,32 @@ This is the buffer SQL strings are sent to. It is stored in the
variable `sql-buffer'. See `sql-help' on how to create such a buffer."
(interactive)
(if (null (buffer-live-p sql-buffer))
- (message "There is no SQLi buffer.")
+ (message "%s has no SQLi buffer set." (buffer-name (current-buffer)))
(if (null (get-buffer-process sql-buffer))
(message "Buffer %s has no process." (buffer-name sql-buffer))
(message "Current SQLi buffer is %s." (buffer-name sql-buffer)))))
+(defun sql-make-alternate-buffer-name ()
+ "Return a string that can be used to rename a SQLi buffer.
+
+This is used to set `sql-alternate-buffer-name' within
+`sql-interactive-mode'."
+ (concat (if (string= "" sql-user)
+ (if (string= "" user-login-name)
+ ()
+ (concat user-login-name "/"))
+ (concat sql-user "/"))
+ (if (string= "" sql-database)
+ (if (string= "" sql-server)
+ system-name
+ sql-server)
+ sql-database)))
+
+(defun sql-rename-buffer ()
+ "Renames a SQLi buffer."
+ (interactive)
+ (rename-buffer (format "*SQL: %s*" sql-alternate-buffer-name) t))
+
(defun sql-copy-column ()
"Copy current column to the end of buffer.
Inserts SELECT or commas if appropriate."
@@ -735,6 +804,17 @@ Inserts SELECT or commas if appropriate."
(point))))
(sql-send-region start end)))
+(defun sql-send-paragraph ()
+ "Send the current paragraph to the SQL process."
+ (interactive)
+ (let ((start (save-excursion
+ (backward-paragraph)
+ (point)))
+ (end (save-excursion
+ (forward-paragraph)
+ (point))))
+ (sql-send-region start end)))
+
(defun sql-send-buffer ()
"Send the buffer contents to the SQL process."
(interactive)
@@ -770,8 +850,8 @@ When you put a buffer in SQL mode, the buffer stores the last SQLi
buffer created as its destination in the variable `sql-buffer'. This
will be the buffer \\[sql-send-region] sends the region to. If this
SQLi buffer is killed, \\[sql-send-region] is no longer able to
-determine where the strings should be sent to. You can change the
-value of `sql-buffer' using \\[sql-change-sqli-buffer].
+determine where the strings should be sent to. You can set the
+value of `sql-buffer' using \\[sql-set-sqli-buffer].
For information on how to create multiple SQLi buffers, see
`sql-interactive-mode'."
@@ -786,9 +866,14 @@ For information on how to create multiple SQLi buffers, see
nil t ((95 . "w") (46 . "w"))))
(make-local-variable 'comment-start)
(setq comment-start "--")
- ;; The following will make each buffer in sql-mode remeber the
- ;; "current" SQLi buffer.
+ ;; Make each buffer in sql-mode remeber the "current" SQLi buffer.
(make-local-variable 'sql-buffer)
+ ;; Make `sql-send-paragraph' work on paragraphs that contain indented
+ ;; lines.
+ (make-local-variable 'paragraph-separate)
+ (make-local-variable 'paragraph-start)
+ (setq paragraph-separate "[\f]*$"
+ paragraph-start "[\n\f]")
(setq local-abbrev-table sql-mode-abbrev-table)
(setq abbrev-all-caps 1)
(run-hooks 'sql-mode-hook))
@@ -844,8 +929,11 @@ hooks on `comint-input-filter-functions' are run. After each SQL
interpreter output, the hooks on `comint-output-filter-functions' are
run.
-Variable `comint-input-ring-file-name' controls the initialisation of
-the input ring history.
+Variable `sql-input-ring-file-name' controls the initialisation of the
+input ring history. `comint-input-ring-file-name' is temporarily bound
+to `sql-input-ring-file-name' and `comint-input-ring-separator' is
+temporarily bound to `sql-input-ring-separator' when reading the input
+history.
Variables `comint-output-filter-functions', a hook, and
`comint-scroll-to-bottom-on-input' and
@@ -856,11 +944,10 @@ If you want to make SQL buffers limited in length, add the function
`comint-truncate-buffer' to `comint-output-filter-functions'.
Here is an example for your .emacs file. It keeps the SQLi buffer a
-certain length and stores all inputs in an input-ring file.
+certain length.
\(add-hook 'sql-interactive-mode-hook
\(function (lambda ()
- \(setq comint-input-ring-file-name \"~/.sql_history\")
\(setq comint-output-filter-functions 'comint-truncate-buffer))))
Here is another example. It will always put point back to the statement
@@ -882,25 +969,48 @@ you entered, right above the output it created.
;; is disabled for interactive buffers.
(setq font-lock-defaults '(sql-mode-font-lock-keywords
t t ((95 . "w") (46 . "w"))))
+ ;; Enable commenting and uncommenting of the region.
(make-local-variable 'comment-start)
(setq comment-start "--")
+ ;; Abbreviation table init and case-insensitive. It is not activatet
+ ;; by default.
(setq local-abbrev-table sql-mode-abbrev-table)
(setq abbrev-all-caps 1)
+ ;; Exiting the process will call sql-stop.
(set-process-sentinel (get-buffer-process sql-buffer) 'sql-stop)
+ ;; Make input-ring stuff buffer local so that people who want a
+ ;; different history file for each buffer/process/client/whatever can
+ ;; change separator and file-name on the sql-interactive-mode-hook.
+ (make-local-variable 'sql-input-ring-separator)
+ (make-local-variable 'sql-input-ring-file-name)
+ ;; Create a usefull name for renaming this buffer later.
+ (make-local-variable 'sql-alternate-buffer-name)
+ (setq sql-alternate-buffer-name (sql-make-alternate-buffer-name))
+ ;; User stuff.
(run-hooks 'sql-interactive-mode-hook)
- ;; calling the hook before calling comint-read-input-ring allows users
+ ;; Calling the hook before calling comint-read-input-ring allows users
;; to set comint-input-ring-file-name in sql-interactive-mode-hook.
- (comint-read-input-ring t))
+ ;; While reading the history, file-name and history are rebound...
+ (let ((comint-input-ring-file-name sql-input-ring-file-name)
+ (comint-input-ring-separator sql-input-ring-separator))
+ (comint-read-input-ring t)))
(defun sql-stop (process event)
"Called when the SQL process is stopped.
-Writes the input history to a history file using `comint-write-input-ring'
-and inserts a short message in the SQL buffer.
+Writes the input history to a history file using
+`comint-write-input-ring' and inserts a short message in the SQL buffer.
+`comint-comint-input-ring-file-name' is temporarily bound to
+`sql-input-ring-file-name' and `comint-input-ring-separator' is
+temporarily bound to `sql-input-ring-separator'.
This function is a sentinel watching the SQL interpreter process.
Sentinels will always get the two parameters PROCESS and EVENT."
- (comint-write-input-ring)
+ ;; Write history.
+ ;; While reading the history, file-name and history are rebound...
+ (let ((comint-input-ring-file-name sql-input-ring-file-name)
+ (comint-input-ring-separator sql-input-ring-separator))
+ (comint-write-input-ring))
(if (buffer-live-p sql-buffer)
(insert (format "\nProcess %s %s\n" process event))))