summaryrefslogtreecommitdiff
path: root/ext/pspell
diff options
context:
space:
mode:
Diffstat (limited to 'ext/pspell')
-rw-r--r--ext/pspell/CREDITS2
-rw-r--r--ext/pspell/README51
-rw-r--r--ext/pspell/config.m445
-rw-r--r--ext/pspell/config.w3216
-rw-r--r--ext/pspell/php_pspell.h32
-rw-r--r--ext/pspell/pspell.c935
-rw-r--r--ext/pspell/pspell.dsp112
-rw-r--r--ext/pspell/tests/001.phpt116
-rw-r--r--ext/pspell/tests/002.phpt42
-rw-r--r--ext/pspell/tests/003.phpt39
-rw-r--r--ext/pspell/tests/004.phpt35
-rw-r--r--ext/pspell/tests/005.phpt46
-rw-r--r--ext/pspell/tests/wordlist.txt5
13 files changed, 1476 insertions, 0 deletions
diff --git a/ext/pspell/CREDITS b/ext/pspell/CREDITS
new file mode 100644
index 0000000..a7919d4
--- /dev/null
+++ b/ext/pspell/CREDITS
@@ -0,0 +1,2 @@
+Pspell
+Vlad Krupin
diff --git a/ext/pspell/README b/ext/pspell/README
new file mode 100644
index 0000000..2e68a2c
--- /dev/null
+++ b/ext/pspell/README
@@ -0,0 +1,51 @@
+README file for pspell (spellchecker) module for PHP
+----------------------------------------------------
+
+The latest release of pspell is always available from
+
+ http://aspell.net/
+
+This module was developed and tested with aspell-.31.1 and pspell-.11.0.2,
+although slightly earlier (and hopefully later) versions of those libraries
+should work as well.
+
+General notes
+-------------
+
+Please, note that somewhere around version .27.x (I believe) aspell stopped
+working with the aspell module for php. This is due to the fact that the
+author changed things around a bit, and suggested that people link to pspell
+in the future rather than to aspell. That's exactly what this module is for.
+It has the same basic functionality as aspell (and more features are being
+added). I did not want to modify existing aspell module, because it would
+break things for those who are using older aspell, or result in very ugly code.
+Rather, I wrote a new module - pspell.
+
+
+Building pspell on a Unix system
+--------------------------------
+
+In order to use pspell, you need to have *both* aspell and pspell libraries
+installed, and they have to be compatible with each other. Get the latest
+release of both at the URL given above.
+
+I expect the libraries to be in /usr/local (that's the default when you
+cofigure pspell and aspell with their 'configure' scripts woth no parameters).
+If that location is different, please specify it in --with-pspell=PATH, where
+PATH is the path you specified for pspell libraries.
+
+Notes on personalized dictionaries
+----------------------------------
+
+It is now possible to have personalized dictionaries with pspell. This is
+achieved by opening a dictionary with pspell_new_personal() rather than
+pspell_new() and specifying the custom dictionary as the first parameter.
+The parameter probably should begin with '/' because otherwise it will
+be relative to $HOME, which is probably set to "/root". This is probably not
+what you want because you need write access for nobody to be able to write
+to a dictionary. Once the dictionary is open, you can use
+pspell_add_to_personal() to add words to the wordlist and finally
+pspell_save_wordlist() to save it.
+
+Note that at this moment pspell_save_wordlist() will return TRUE, even if
+there was an error. This will be changed once pspell library is fixed. \ No newline at end of file
diff --git a/ext/pspell/config.m4 b/ext/pspell/config.m4
new file mode 100644
index 0000000..67e5b27
--- /dev/null
+++ b/ext/pspell/config.m4
@@ -0,0 +1,45 @@
+dnl
+dnl $Id$
+dnl
+
+PHP_ARG_WITH(pspell,for PSPELL support,
+[ --with-pspell[=DIR] Include PSPELL support.
+ GNU Aspell version 0.50.0 or higher required])
+
+if test "$PHP_PSPELL" != "no"; then
+ PHP_NEW_EXTENSION(pspell, pspell.c, $ext_shared)
+ if test "$PHP_PSPELL" != "yes"; then
+ PSPELL_SEARCH_DIRS=$PHP_PSPELL
+ else
+ PSPELL_SEARCH_DIRS="/usr/local /usr"
+ fi
+ for i in $PSPELL_SEARCH_DIRS; do
+ if test -f $i/include/pspell/pspell.h; then
+ PSPELL_DIR=$i
+ PSPELL_INCDIR=$i/include/pspell
+ elif test -f $i/include/pspell.h; then
+ PSPELL_DIR=$i
+ PSPELL_INCDIR=$i/include
+ fi
+ done
+
+ if test -z "$PSPELL_DIR"; then
+ AC_MSG_ERROR(Cannot find pspell)
+ fi
+
+ PSPELL_LIBDIR=$PSPELL_DIR/$PHP_LIBDIR
+
+ PHP_ADD_LIBRARY_WITH_PATH(pspell, $PSPELL_LIBDIR, PSPELL_SHARED_LIBADD)
+
+ dnl Add -laspell to LIBS if it exists
+ PHP_CHECK_LIBRARY(aspell,new_aspell_config,
+ [
+ PHP_ADD_LIBRARY_WITH_PATH(aspell, $PSPELL_LIBDIR, PSPELL_SHARED_LIBADD)
+ ], [], [
+ -L$PSPELL_LIBDIR
+ ])
+
+ PHP_ADD_INCLUDE($PSPELL_INCDIR)
+ PHP_SUBST(PSPELL_SHARED_LIBADD)
+ AC_DEFINE(HAVE_PSPELL,1,[ ])
+fi
diff --git a/ext/pspell/config.w32 b/ext/pspell/config.w32
new file mode 100644
index 0000000..741a4bc
--- /dev/null
+++ b/ext/pspell/config.w32
@@ -0,0 +1,16 @@
+// $Id$
+// vim:ft=javascript
+
+ARG_WITH("pspell", "pspell/aspell (whatever it's called this month) support", "no");
+
+if (PHP_PSPELL != "no") {
+
+ if (CHECK_HEADER_ADD_INCLUDE("pspell.h", "CFLAGS_PSPELL", PHP_PHP_BUILD + "\\include\\pspell;" + PHP_PSPELL) &&
+ CHECK_LIB("aspell*.lib", "pspell", PHP_PSPELL)) {
+ EXTENSION('pspell', 'pspell.c');
+ AC_DEFINE('HAVE_PSPELL', 1);
+ } else {
+ WARNING("pspell not enabled; libraries and headers not found");
+ }
+}
+
diff --git a/ext/pspell/php_pspell.h b/ext/pspell/php_pspell.h
new file mode 100644
index 0000000..b4e97f3
--- /dev/null
+++ b/ext/pspell/php_pspell.h
@@ -0,0 +1,32 @@
+/*
+ +----------------------------------------------------------------------+
+ | PHP Version 5 |
+ +----------------------------------------------------------------------+
+ | Copyright (c) 1997-2013 The PHP Group |
+ +----------------------------------------------------------------------+
+ | This source file is subject to version 3.01 of the PHP license, |
+ | that is bundled with this package in the file LICENSE, and is |
+ | available through the world-wide-web at the following url: |
+ | http://www.php.net/license/3_01.txt |
+ | If you did not receive a copy of the PHP license and are unable to |
+ | obtain it through the world-wide-web, please send a note to |
+ | license@php.net so we can mail you a copy immediately. |
+ +----------------------------------------------------------------------+
+ | Author: Vlad Krupin <phpdevel@echospace.com> |
+ +----------------------------------------------------------------------+
+*/
+
+/* $Id$ */
+
+#ifndef _PSPELL_H
+#define _PSPELL_H
+#if HAVE_PSPELL
+extern zend_module_entry pspell_module_entry;
+#define pspell_module_ptr &pspell_module_entry
+#else
+#define pspell_module_ptr NULL
+#endif
+
+#define phpext_pspell_ptr pspell_module_ptr
+
+#endif /* _PSPELL_H */
diff --git a/ext/pspell/pspell.c b/ext/pspell/pspell.c
new file mode 100644
index 0000000..79a597e
--- /dev/null
+++ b/ext/pspell/pspell.c
@@ -0,0 +1,935 @@
+/*
+ +----------------------------------------------------------------------+
+ | PHP Version 5 |
+ +----------------------------------------------------------------------+
+ | Copyright (c) 1997-2013 The PHP Group |
+ +----------------------------------------------------------------------+
+ | This source file is subject to version 3.01 of the PHP license, |
+ | that is bundled with this package in the file LICENSE, and is |
+ | available through the world-wide-web at the following url: |
+ | http://www.php.net/license/3_01.txt |
+ | If you did not receive a copy of the PHP license and are unable to |
+ | obtain it through the world-wide-web, please send a note to |
+ | license@php.net so we can mail you a copy immediately. |
+ +----------------------------------------------------------------------+
+ | Author: Vlad Krupin <phpdevel@echospace.com> |
+ +----------------------------------------------------------------------+
+*/
+
+/* $Id$ */
+
+#define IS_EXT_MODULE
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include "php.h"
+
+#include <stdlib.h>
+#include <ctype.h>
+#include <stdio.h>
+
+#if HAVE_PSPELL
+
+/* this will enforce compatibility in .12 version (broken after .11.2) */
+#define USE_ORIGINAL_MANAGER_FUNCS
+
+#include "php_pspell.h"
+#include <pspell.h>
+#include "ext/standard/info.h"
+
+#define PSPELL_FAST 1L
+#define PSPELL_NORMAL 2L
+#define PSPELL_BAD_SPELLERS 3L
+#define PSPELL_SPEED_MASK_INTERNAL 3L
+#define PSPELL_RUN_TOGETHER 8L
+
+/* Largest ignored word can be 999 characters (this seems sane enough),
+ * and it takes 3 bytes to represent that (see pspell_config_ignore)
+ */
+#define PSPELL_LARGEST_WORD 3
+
+static PHP_MINIT_FUNCTION(pspell);
+static PHP_MINFO_FUNCTION(pspell);
+static PHP_FUNCTION(pspell_new);
+static PHP_FUNCTION(pspell_new_personal);
+static PHP_FUNCTION(pspell_new_config);
+static PHP_FUNCTION(pspell_check);
+static PHP_FUNCTION(pspell_suggest);
+static PHP_FUNCTION(pspell_store_replacement);
+static PHP_FUNCTION(pspell_add_to_personal);
+static PHP_FUNCTION(pspell_add_to_session);
+static PHP_FUNCTION(pspell_clear_session);
+static PHP_FUNCTION(pspell_save_wordlist);
+static PHP_FUNCTION(pspell_config_create);
+static PHP_FUNCTION(pspell_config_runtogether);
+static PHP_FUNCTION(pspell_config_mode);
+static PHP_FUNCTION(pspell_config_ignore);
+static PHP_FUNCTION(pspell_config_personal);
+static PHP_FUNCTION(pspell_config_dict_dir);
+static PHP_FUNCTION(pspell_config_data_dir);
+static PHP_FUNCTION(pspell_config_repl);
+static PHP_FUNCTION(pspell_config_save_repl);
+
+/* {{{ arginfo */
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_new, 0, 0, 1)
+ ZEND_ARG_INFO(0, language)
+ ZEND_ARG_INFO(0, spelling)
+ ZEND_ARG_INFO(0, jargon)
+ ZEND_ARG_INFO(0, encoding)
+ ZEND_ARG_INFO(0, mode)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_new_personal, 0, 0, 2)
+ ZEND_ARG_INFO(0, personal)
+ ZEND_ARG_INFO(0, language)
+ ZEND_ARG_INFO(0, spelling)
+ ZEND_ARG_INFO(0, jargon)
+ ZEND_ARG_INFO(0, encoding)
+ ZEND_ARG_INFO(0, mode)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_new_config, 0, 0, 1)
+ ZEND_ARG_INFO(0, config)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_check, 0, 0, 2)
+ ZEND_ARG_INFO(0, pspell)
+ ZEND_ARG_INFO(0, word)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_suggest, 0, 0, 2)
+ ZEND_ARG_INFO(0, pspell)
+ ZEND_ARG_INFO(0, word)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_store_replacement, 0, 0, 3)
+ ZEND_ARG_INFO(0, pspell)
+ ZEND_ARG_INFO(0, misspell)
+ ZEND_ARG_INFO(0, correct)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_add_to_personal, 0, 0, 2)
+ ZEND_ARG_INFO(0, pspell)
+ ZEND_ARG_INFO(0, word)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_add_to_session, 0, 0, 2)
+ ZEND_ARG_INFO(0, pspell)
+ ZEND_ARG_INFO(0, word)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_clear_session, 0, 0, 1)
+ ZEND_ARG_INFO(0, pspell)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_save_wordlist, 0, 0, 1)
+ ZEND_ARG_INFO(0, pspell)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_create, 0, 0, 1)
+ ZEND_ARG_INFO(0, language)
+ ZEND_ARG_INFO(0, spelling)
+ ZEND_ARG_INFO(0, jargon)
+ ZEND_ARG_INFO(0, encoding)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_runtogether, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, runtogether)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_mode, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, mode)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_ignore, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, ignore)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_personal, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, personal)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_dict_dir, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, directory)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_data_dir, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, directory)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_repl, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, repl)
+ZEND_END_ARG_INFO()
+
+ZEND_BEGIN_ARG_INFO_EX(arginfo_pspell_config_save_repl, 0, 0, 2)
+ ZEND_ARG_INFO(0, conf)
+ ZEND_ARG_INFO(0, save)
+ZEND_END_ARG_INFO()
+/* }}} */
+
+/* {{{ pspell_functions[]
+ */
+static const zend_function_entry pspell_functions[] = {
+ PHP_FE(pspell_new, arginfo_pspell_new)
+ PHP_FE(pspell_new_personal, arginfo_pspell_new_personal)
+ PHP_FE(pspell_new_config, arginfo_pspell_new_config)
+ PHP_FE(pspell_check, arginfo_pspell_check)
+ PHP_FE(pspell_suggest, arginfo_pspell_suggest)
+ PHP_FE(pspell_store_replacement, arginfo_pspell_store_replacement)
+ PHP_FE(pspell_add_to_personal, arginfo_pspell_add_to_personal)
+ PHP_FE(pspell_add_to_session, arginfo_pspell_add_to_session)
+ PHP_FE(pspell_clear_session, arginfo_pspell_clear_session)
+ PHP_FE(pspell_save_wordlist, arginfo_pspell_save_wordlist)
+ PHP_FE(pspell_config_create, arginfo_pspell_config_create)
+ PHP_FE(pspell_config_runtogether, arginfo_pspell_config_runtogether)
+ PHP_FE(pspell_config_mode, arginfo_pspell_config_mode)
+ PHP_FE(pspell_config_ignore, arginfo_pspell_config_ignore)
+ PHP_FE(pspell_config_personal, arginfo_pspell_config_personal)
+ PHP_FE(pspell_config_dict_dir, arginfo_pspell_config_dict_dir)
+ PHP_FE(pspell_config_data_dir, arginfo_pspell_config_data_dir)
+ PHP_FE(pspell_config_repl, arginfo_pspell_config_repl)
+ PHP_FE(pspell_config_save_repl, arginfo_pspell_config_save_repl)
+ PHP_FE_END
+};
+/* }}} */
+
+static int le_pspell, le_pspell_config;
+
+zend_module_entry pspell_module_entry = {
+ STANDARD_MODULE_HEADER,
+ "pspell", pspell_functions, PHP_MINIT(pspell), NULL, NULL, NULL, PHP_MINFO(pspell), NO_VERSION_YET, STANDARD_MODULE_PROPERTIES
+};
+
+#ifdef COMPILE_DL_PSPELL
+ZEND_GET_MODULE(pspell)
+#endif
+
+static void php_pspell_close(zend_rsrc_list_entry *rsrc TSRMLS_DC)
+{
+ PspellManager *manager = (PspellManager *)rsrc->ptr;
+
+ delete_pspell_manager(manager);
+}
+
+static void php_pspell_close_config(zend_rsrc_list_entry *rsrc TSRMLS_DC)
+{
+ PspellConfig *config = (PspellConfig *)rsrc->ptr;
+
+ delete_pspell_config(config);
+}
+
+#define PSPELL_FETCH_CONFIG \
+ config = (PspellConfig *) zend_list_find(conf, &type); \
+ if (config == NULL || type != le_pspell_config) { \
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "%ld is not a PSPELL config index", conf); \
+ RETURN_FALSE; \
+ } \
+
+#define PSPELL_FETCH_MANAGER \
+ manager = (PspellManager *) zend_list_find(scin, &type); \
+ if (!manager || type != le_pspell) { \
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "%ld is not a PSPELL result index", scin); \
+ RETURN_FALSE; \
+ } \
+
+/* {{{ PHP_MINIT_FUNCTION
+ */
+static PHP_MINIT_FUNCTION(pspell)
+{
+ REGISTER_LONG_CONSTANT("PSPELL_FAST", PSPELL_FAST, CONST_PERSISTENT | CONST_CS);
+ REGISTER_LONG_CONSTANT("PSPELL_NORMAL", PSPELL_NORMAL, CONST_PERSISTENT | CONST_CS);
+ REGISTER_LONG_CONSTANT("PSPELL_BAD_SPELLERS", PSPELL_BAD_SPELLERS, CONST_PERSISTENT | CONST_CS);
+ REGISTER_LONG_CONSTANT("PSPELL_RUN_TOGETHER", PSPELL_RUN_TOGETHER, CONST_PERSISTENT | CONST_CS);
+ le_pspell = zend_register_list_destructors_ex(php_pspell_close, NULL, "pspell", module_number);
+ le_pspell_config = zend_register_list_destructors_ex(php_pspell_close_config, NULL, "pspell config", module_number);
+ return SUCCESS;
+}
+/* }}} */
+
+/* {{{ proto int pspell_new(string language [, string spelling [, string jargon [, string encoding [, int mode]]]])
+ Load a dictionary */
+static PHP_FUNCTION(pspell_new)
+{
+ char *language, *spelling = NULL, *jargon = NULL, *encoding = NULL;
+ int language_len, spelling_len = 0, jargon_len = 0, encoding_len = 0;
+ long mode = 0L, speed = 0L;
+ int argc = ZEND_NUM_ARGS();
+ int ind;
+
+#ifdef PHP_WIN32
+ TCHAR aspell_dir[200];
+ TCHAR data_dir[220];
+ TCHAR dict_dir[220];
+ HKEY hkey;
+ DWORD dwType,dwLen;
+#endif
+
+ PspellCanHaveError *ret;
+ PspellManager *manager;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(argc TSRMLS_CC, "s|sssl", &language, &language_len, &spelling, &spelling_len,
+ &jargon, &jargon_len, &encoding, &encoding_len, &mode) == FAILURE) {
+ return;
+ }
+
+ config = new_pspell_config();
+
+#ifdef PHP_WIN32
+ /* If aspell was installed using installer, we should have a key
+ * pointing to the location of the dictionaries
+ */
+ if (0 == RegOpenKey(HKEY_LOCAL_MACHINE, "Software\\Aspell", &hkey)) {
+ LONG result;
+ dwLen = sizeof(aspell_dir) - 1;
+ result = RegQueryValueEx(hkey, "", NULL, &dwType, (LPBYTE)&aspell_dir, &dwLen);
+ RegCloseKey(hkey);
+ if (result == ERROR_SUCCESS) {
+ strlcpy(data_dir, aspell_dir, sizeof(data_dir));
+ strlcat(data_dir, "\\data", sizeof(data_dir));
+ strlcpy(dict_dir, aspell_dir, sizeof(dict_dir));
+ strlcat(dict_dir, "\\dict", sizeof(dict_dir));
+
+ pspell_config_replace(config, "data-dir", data_dir);
+ pspell_config_replace(config, "dict-dir", dict_dir);
+ }
+ }
+#endif
+
+ pspell_config_replace(config, "language-tag", language);
+
+ if (spelling_len) {
+ pspell_config_replace(config, "spelling", spelling);
+ }
+
+ if (jargon_len) {
+ pspell_config_replace(config, "jargon", jargon);
+ }
+
+ if (encoding_len) {
+ pspell_config_replace(config, "encoding", encoding);
+ }
+
+ if (argc > 4) {
+ speed = mode & PSPELL_SPEED_MASK_INTERNAL;
+
+ /* First check what mode we want (how many suggestions) */
+ if (speed == PSPELL_FAST) {
+ pspell_config_replace(config, "sug-mode", "fast");
+ } else if (speed == PSPELL_NORMAL) {
+ pspell_config_replace(config, "sug-mode", "normal");
+ } else if (speed == PSPELL_BAD_SPELLERS) {
+ pspell_config_replace(config, "sug-mode", "bad-spellers");
+ }
+
+ /* Then we see if run-together words should be treated as valid components */
+ if (mode & PSPELL_RUN_TOGETHER) {
+ pspell_config_replace(config, "run-together", "true");
+ }
+ }
+
+ ret = new_pspell_manager(config);
+ delete_pspell_config(config);
+
+ if (pspell_error_number(ret) != 0) {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "PSPELL couldn't open the dictionary. reason: %s", pspell_error_message(ret));
+ delete_pspell_can_have_error(ret);
+ RETURN_FALSE;
+ }
+
+ manager = to_pspell_manager(ret);
+ ind = zend_list_insert(manager, le_pspell TSRMLS_CC);
+ RETURN_LONG(ind);
+}
+/* }}} */
+
+/* {{{ proto int pspell_new_personal(string personal, string language [, string spelling [, string jargon [, string encoding [, int mode]]]])
+ Load a dictionary with a personal wordlist*/
+static PHP_FUNCTION(pspell_new_personal)
+{
+ char *personal, *language, *spelling = NULL, *jargon = NULL, *encoding = NULL;
+ int personal_len, language_len, spelling_len = 0, jargon_len = 0, encoding_len = 0;
+ long mode = 0L, speed = 0L;
+ int argc = ZEND_NUM_ARGS();
+ int ind;
+
+#ifdef PHP_WIN32
+ TCHAR aspell_dir[200];
+ TCHAR data_dir[220];
+ TCHAR dict_dir[220];
+ HKEY hkey;
+ DWORD dwType,dwLen;
+#endif
+
+ PspellCanHaveError *ret;
+ PspellManager *manager;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(argc TSRMLS_CC, "ps|sssl", &personal, &personal_len, &language, &language_len,
+ &spelling, &spelling_len, &jargon, &jargon_len, &encoding, &encoding_len, &mode) == FAILURE) {
+ return;
+ }
+
+ config = new_pspell_config();
+
+#ifdef PHP_WIN32
+ /* If aspell was installed using installer, we should have a key
+ * pointing to the location of the dictionaries
+ */
+ if (0 == RegOpenKey(HKEY_LOCAL_MACHINE, "Software\\Aspell", &hkey)) {
+ LONG result;
+ dwLen = sizeof(aspell_dir) - 1;
+ result = RegQueryValueEx(hkey, "", NULL, &dwType, (LPBYTE)&aspell_dir, &dwLen);
+ RegCloseKey(hkey);
+ if (result == ERROR_SUCCESS) {
+ strlcpy(data_dir, aspell_dir, sizeof(data_dir));
+ strlcat(data_dir, "\\data", sizeof(data_dir));
+ strlcpy(dict_dir, aspell_dir, sizeof(dict_dir));
+ strlcat(dict_dir, "\\dict", sizeof(dict_dir));
+
+ pspell_config_replace(config, "data-dir", data_dir);
+ pspell_config_replace(config, "dict-dir", dict_dir);
+ }
+ }
+#endif
+
+ if (php_check_open_basedir(personal TSRMLS_CC)) {
+ delete_pspell_config(config);
+ RETURN_FALSE;
+ }
+
+ pspell_config_replace(config, "personal", personal);
+ pspell_config_replace(config, "save-repl", "false");
+
+ pspell_config_replace(config, "language-tag", language);
+
+ if (spelling_len) {
+ pspell_config_replace(config, "spelling", spelling);
+ }
+
+ if (jargon_len) {
+ pspell_config_replace(config, "jargon", jargon);
+ }
+
+ if (encoding_len) {
+ pspell_config_replace(config, "encoding", encoding);
+ }
+
+ if (argc > 5) {
+ speed = mode & PSPELL_SPEED_MASK_INTERNAL;
+
+ /* First check what mode we want (how many suggestions) */
+ if (speed == PSPELL_FAST) {
+ pspell_config_replace(config, "sug-mode", "fast");
+ } else if (speed == PSPELL_NORMAL) {
+ pspell_config_replace(config, "sug-mode", "normal");
+ } else if (speed == PSPELL_BAD_SPELLERS) {
+ pspell_config_replace(config, "sug-mode", "bad-spellers");
+ }
+
+ /* Then we see if run-together words should be treated as valid components */
+ if (mode & PSPELL_RUN_TOGETHER) {
+ pspell_config_replace(config, "run-together", "true");
+ }
+ }
+
+ ret = new_pspell_manager(config);
+ delete_pspell_config(config);
+
+ if (pspell_error_number(ret) != 0) {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "PSPELL couldn't open the dictionary. reason: %s", pspell_error_message(ret));
+ delete_pspell_can_have_error(ret);
+ RETURN_FALSE;
+ }
+
+ manager = to_pspell_manager(ret);
+ ind = zend_list_insert(manager, le_pspell TSRMLS_CC);
+ RETURN_LONG(ind);
+}
+/* }}} */
+
+/* {{{ proto int pspell_new_config(int config)
+ Load a dictionary based on the given config */
+static PHP_FUNCTION(pspell_new_config)
+{
+ int type, ind;
+ long conf;
+ PspellCanHaveError *ret;
+ PspellManager *manager;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &conf) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ ret = new_pspell_manager(config);
+
+ if (pspell_error_number(ret) != 0) {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "PSPELL couldn't open the dictionary. reason: %s", pspell_error_message(ret));
+ delete_pspell_can_have_error(ret);
+ RETURN_FALSE;
+ }
+
+ manager = to_pspell_manager(ret);
+ ind = zend_list_insert(manager, le_pspell TSRMLS_CC);
+ RETURN_LONG(ind);
+}
+/* }}} */
+
+/* {{{ proto bool pspell_check(int pspell, string word)
+ Returns true if word is valid */
+static PHP_FUNCTION(pspell_check)
+{
+ int type, word_len;
+ long scin;
+ char *word;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ls", &scin, &word, &word_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ if (pspell_manager_check(manager, word)) {
+ RETURN_TRUE;
+ } else {
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto array pspell_suggest(int pspell, string word)
+ Returns array of suggestions */
+static PHP_FUNCTION(pspell_suggest)
+{
+ long scin;
+ char *word;
+ int word_len;
+ PspellManager *manager;
+ int type;
+ const PspellWordList *wl;
+ const char *sug;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ls", &scin, &word, &word_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ array_init(return_value);
+
+ wl = pspell_manager_suggest(manager, word);
+ if (wl) {
+ PspellStringEmulation *els = pspell_word_list_elements(wl);
+ while ((sug = pspell_string_emulation_next(els)) != 0) {
+ add_next_index_string(return_value,(char *)sug,1);
+ }
+ delete_pspell_string_emulation(els);
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "PSPELL had a problem. details: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto bool pspell_store_replacement(int pspell, string misspell, string correct)
+ Notify the dictionary of a user-selected replacement */
+static PHP_FUNCTION(pspell_store_replacement)
+{
+ int type, miss_len, corr_len;
+ long scin;
+ char *miss, *corr;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lss", &scin, &miss, &miss_len, &corr, &corr_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ pspell_manager_store_replacement(manager, miss, corr);
+ if (pspell_manager_error_number(manager) == 0) {
+ RETURN_TRUE;
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "pspell_store_replacement() gave error: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto bool pspell_add_to_personal(int pspell, string word)
+ Adds a word to a personal list */
+static PHP_FUNCTION(pspell_add_to_personal)
+{
+ int type, word_len;
+ long scin;
+ char *word;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ls", &scin, &word, &word_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ /*If the word is empty, we have to return; otherwise we'll segfault! ouch!*/
+ if (word_len == 0) {
+ RETURN_FALSE;
+ }
+
+ pspell_manager_add_to_personal(manager, word);
+ if (pspell_manager_error_number(manager) == 0) {
+ RETURN_TRUE;
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "pspell_add_to_personal() gave error: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto bool pspell_add_to_session(int pspell, string word)
+ Adds a word to the current session */
+static PHP_FUNCTION(pspell_add_to_session)
+{
+ int type, word_len;
+ long scin;
+ char *word;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ls", &scin, &word, &word_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ /*If the word is empty, we have to return; otherwise we'll segfault! ouch!*/
+ if (word_len == 0) {
+ RETURN_FALSE;
+ }
+
+ pspell_manager_add_to_session(manager, word);
+ if (pspell_manager_error_number(manager) == 0) {
+ RETURN_TRUE;
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "pspell_add_to_session() gave error: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto bool pspell_clear_session(int pspell)
+ Clears the current session */
+static PHP_FUNCTION(pspell_clear_session)
+{
+ int type;
+ long scin;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &scin) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ pspell_manager_clear_session(manager);
+ if (pspell_manager_error_number(manager) == 0) {
+ RETURN_TRUE;
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "pspell_clear_session() gave error: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+}
+/* }}} */
+
+/* {{{ proto bool pspell_save_wordlist(int pspell)
+ Saves the current (personal) wordlist */
+static PHP_FUNCTION(pspell_save_wordlist)
+{
+ int type;
+ long scin;
+ PspellManager *manager;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &scin) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_MANAGER;
+
+ pspell_manager_save_all_word_lists(manager);
+
+ if (pspell_manager_error_number(manager) == 0) {
+ RETURN_TRUE;
+ } else {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "pspell_save_wordlist() gave error: %s", pspell_manager_error_message(manager));
+ RETURN_FALSE;
+ }
+
+}
+/* }}} */
+
+/* {{{ proto int pspell_config_create(string language [, string spelling [, string jargon [, string encoding]]])
+ Create a new config to be used later to create a manager */
+static PHP_FUNCTION(pspell_config_create)
+{
+ char *language, *spelling = NULL, *jargon = NULL, *encoding = NULL;
+ int language_len, spelling_len = 0, jargon_len = 0, encoding_len = 0;
+ int ind;
+ PspellConfig *config;
+
+#ifdef PHP_WIN32
+ TCHAR aspell_dir[200];
+ TCHAR data_dir[220];
+ TCHAR dict_dir[220];
+ HKEY hkey;
+ DWORD dwType,dwLen;
+#endif
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|sss", &language, &language_len, &spelling, &spelling_len,
+ &jargon, &jargon_len, &encoding, &encoding_len) == FAILURE) {
+ return;
+ }
+
+ config = new_pspell_config();
+
+#ifdef PHP_WIN32
+ /* If aspell was installed using installer, we should have a key
+ * pointing to the location of the dictionaries
+ */
+ if (0 == RegOpenKey(HKEY_LOCAL_MACHINE, "Software\\Aspell", &hkey)) {
+ LONG result;
+ dwLen = sizeof(aspell_dir) - 1;
+ result = RegQueryValueEx(hkey, "", NULL, &dwType, (LPBYTE)&aspell_dir, &dwLen);
+ RegCloseKey(hkey);
+ if (result == ERROR_SUCCESS) {
+ strlcpy(data_dir, aspell_dir, sizeof(data_dir));
+ strlcat(data_dir, "\\data", sizeof(data_dir));
+ strlcpy(dict_dir, aspell_dir, sizeof(dict_dir));
+ strlcat(dict_dir, "\\dict", sizeof(dict_dir));
+
+ pspell_config_replace(config, "data-dir", data_dir);
+ pspell_config_replace(config, "dict-dir", dict_dir);
+ }
+ }
+#endif
+
+ pspell_config_replace(config, "language-tag", language);
+
+ if (spelling_len) {
+ pspell_config_replace(config, "spelling", spelling);
+ }
+
+ if (jargon_len) {
+ pspell_config_replace(config, "jargon", jargon);
+ }
+
+ if (encoding_len) {
+ pspell_config_replace(config, "encoding", encoding);
+ }
+
+ /* By default I do not want to write anything anywhere because it'll try to write to $HOME
+ which is not what we want */
+ pspell_config_replace(config, "save-repl", "false");
+
+ ind = zend_list_insert(config, le_pspell_config TSRMLS_CC);
+ RETURN_LONG(ind);
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_runtogether(int conf, bool runtogether)
+ Consider run-together words as valid components */
+static PHP_FUNCTION(pspell_config_runtogether)
+{
+ int type;
+ long conf;
+ zend_bool runtogether;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lb", &conf, &runtogether) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ pspell_config_replace(config, "run-together", runtogether ? "true" : "false");
+
+ RETURN_TRUE;
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_mode(int conf, long mode)
+ Select mode for config (PSPELL_FAST, PSPELL_NORMAL or PSPELL_BAD_SPELLERS) */
+static PHP_FUNCTION(pspell_config_mode)
+{
+ int type;
+ long conf, mode;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ll", &conf, &mode) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ /* First check what mode we want (how many suggestions) */
+ if (mode == PSPELL_FAST) {
+ pspell_config_replace(config, "sug-mode", "fast");
+ } else if (mode == PSPELL_NORMAL) {
+ pspell_config_replace(config, "sug-mode", "normal");
+ } else if (mode == PSPELL_BAD_SPELLERS) {
+ pspell_config_replace(config, "sug-mode", "bad-spellers");
+ }
+
+ RETURN_TRUE;
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_ignore(int conf, int ignore)
+ Ignore words <= n chars */
+static PHP_FUNCTION(pspell_config_ignore)
+{
+ int type;
+ char ignore_str[MAX_LENGTH_OF_LONG + 1];
+ long conf, ignore = 0L;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ll", &conf, &ignore) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ snprintf(ignore_str, sizeof(ignore_str), "%ld", ignore);
+
+ pspell_config_replace(config, "ignore", ignore_str);
+ RETURN_TRUE;
+}
+/* }}} */
+
+static void pspell_config_path(INTERNAL_FUNCTION_PARAMETERS, char *option)
+{
+ int type;
+ long conf;
+ char *value;
+ int value_len;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lp", &conf, &value, &value_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ if (php_check_open_basedir(value TSRMLS_CC)) {
+ RETURN_FALSE;
+ }
+
+ pspell_config_replace(config, option, value);
+
+ RETURN_TRUE;
+}
+
+/* {{{ proto bool pspell_config_personal(int conf, string personal)
+ Use a personal dictionary for this config */
+static PHP_FUNCTION(pspell_config_personal)
+{
+ pspell_config_path(INTERNAL_FUNCTION_PARAM_PASSTHRU, "personal");
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_dict_dir(int conf, string directory)
+ location of the main word list */
+static PHP_FUNCTION(pspell_config_dict_dir)
+{
+ pspell_config_path(INTERNAL_FUNCTION_PARAM_PASSTHRU, "dict-dir");
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_data_dir(int conf, string directory)
+ location of language data files */
+static PHP_FUNCTION(pspell_config_data_dir)
+{
+ pspell_config_path(INTERNAL_FUNCTION_PARAM_PASSTHRU, "data-dir");
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_repl(int conf, string repl)
+ Use a personal dictionary with replacement pairs for this config */
+static PHP_FUNCTION(pspell_config_repl)
+{
+ int type;
+ long conf;
+ char *repl;
+ int repl_len;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lp", &conf, &repl, &repl_len) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ pspell_config_replace(config, "save-repl", "true");
+
+ if (php_check_open_basedir(repl TSRMLS_CC)) {
+ RETURN_FALSE;
+ }
+
+ pspell_config_replace(config, "repl", repl);
+
+ RETURN_TRUE;
+}
+/* }}} */
+
+/* {{{ proto bool pspell_config_save_repl(int conf, bool save)
+ Save replacement pairs when personal list is saved for this config */
+static PHP_FUNCTION(pspell_config_save_repl)
+{
+ int type;
+ long conf;
+ zend_bool save;
+ PspellConfig *config;
+
+ if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lb", &conf, &save) == FAILURE) {
+ return;
+ }
+
+ PSPELL_FETCH_CONFIG;
+
+ pspell_config_replace(config, "save-repl", save ? "true" : "false");
+
+ RETURN_TRUE;
+}
+/* }}} */
+
+/* {{{ PHP_MINFO_FUNCTION
+ */
+static PHP_MINFO_FUNCTION(pspell)
+{
+ php_info_print_table_start();
+ php_info_print_table_row(2, "PSpell Support", "enabled");
+ php_info_print_table_end();
+}
+/* }}} */
+
+#endif
+
+/*
+ * Local variables:
+ * tab-width: 4
+ * c-basic-offset: 4
+ * End:
+ * vim600: sw=4 ts=4 fdm=marker
+ * vim<600: sw=4 ts=4
+ */
diff --git a/ext/pspell/pspell.dsp b/ext/pspell/pspell.dsp
new file mode 100644
index 0000000..f310099
--- /dev/null
+++ b/ext/pspell/pspell.dsp
@@ -0,0 +1,112 @@
+# Microsoft Developer Studio Project File - Name="pspell" - Package Owner=<4>
+# Microsoft Developer Studio Generated Build File, Format Version 6.00
+# ** DO NOT EDIT **
+
+# TARGTYPE "Win32 (x86) Dynamic-Link Library" 0x0102
+
+CFG=pspell - Win32 Debug_TS
+!MESSAGE This is not a valid makefile. To build this project using NMAKE,
+!MESSAGE use the Export Makefile command and run
+!MESSAGE
+!MESSAGE NMAKE /f "pspell.mak".
+!MESSAGE
+!MESSAGE You can specify a configuration when running NMAKE
+!MESSAGE by defining the macro CFG on the command line. For example:
+!MESSAGE
+!MESSAGE NMAKE /f "pspell.mak" CFG="pspell - Win32 Debug_TS"
+!MESSAGE
+!MESSAGE Possible choices for configuration are:
+!MESSAGE
+!MESSAGE "pspell - Win32 Debug_TS" (based on "Win32 (x86) Dynamic-Link Library")
+!MESSAGE "pspell - Win32 Release_TS" (based on "Win32 (x86) Dynamic-Link Library")
+!MESSAGE
+
+# Begin Project
+# PROP AllowPerConfigDependencies 0
+# PROP Scc_ProjName ""
+# PROP Scc_LocalPath ""
+CPP=cl.exe
+MTL=midl.exe
+RSC=rc.exe
+
+!IF "$(CFG)" == "pspell - Win32 Debug_TS"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 1
+# PROP BASE Output_Dir "pspell___Win32_Debug_TS"
+# PROP BASE Intermediate_Dir "pspell___Win32_Debug_TS"
+# PROP BASE Ignore_Export_Lib 0
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 1
+# PROP Output_Dir "Debug_TS"
+# PROP Intermediate_Dir "Debug_TS"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /I "..\.." /I "..\..\main" /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\..\zlib" /I "..\..\TSRM" /D ZEND_DEBUG=1 /D "WIN32" /D "ZEND_WIN32" /D "PHP_WIN32" /D "_WINDOWS" /YX /FD /GZ /c
+# ADD CPP /nologo /MTd /W3 /Gm /GX /ZI /Od /I "..\.." /I "..\..\main" /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\..\zlib" /I "..\..\TSRM" /I "..\..\..\aspell\include\pspell" /I "..\..\..\php_build\include\pspell." /D ZEND_DEBUG=1 /D "_DEBUG" /D ZTS=1 /D HAVE_PSPELL=1 /D "COMPILE_DL_PSPELL" /D "WIN32" /D "ZEND_WIN32" /D "PHP_WIN32" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "PSPELL_EXPORTS" /YX /FD /GZ /c
+# ADD BASE MTL /nologo /D "_DEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "_DEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x409 /d "_DEBUG"
+# ADD RSC /l 0x409 /d "_DEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 php5ts_debug.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /debug /machine:I386 /out:"..\..\Debug_TS/php_pspell.dll" /pdbtype:sept /libpath:"..\..\Debug_TS"
+# ADD LINK32 php5ts_debug.lib aspell-15.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /debug /machine:I386 /out:"..\..\Debug_TS/php_pspell.dll" /pdbtype:sept /libpath:"..\..\Debug_TS" /libpath:"..\..\..\aspell\lib"
+
+!ELSEIF "$(CFG)" == "pspell - Win32 Release_TS"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 0
+# PROP BASE Output_Dir "pspell___Win32_Release_TS"
+# PROP BASE Intermediate_Dir "pspell___Win32_Release_TS"
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 0
+# PROP Output_Dir "Release_TS"
+# PROP Intermediate_Dir "Release_TS"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /MT /W3 /GX /O2 /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "PSPELL_EXPORTS" /YX /FD /c
+# ADD CPP /nologo /MT /W3 /GX /O2 /I "..\.." /I "..\..\main" /I "..\..\Zend" /I "..\..\..\bindlib_w32" /I "..\..\..\zlib" /I "..\..\TSRM" /I "..\..\..\aspell\include\pspell" /I "..\..\..\php_build\include\pspell" /D ZEND_DEBUG=0 /D "NDEBUG" /D ZTS=1 /D HAVE_PSPELL=1 /D "COMPILE_DL_PSPELL" /D "WIN32" /D "ZEND_WIN32" /D "PHP_WIN32" /D "_WINDOWS" /D "_MBCS" /D "_USRDLL" /D "PSPELL_EXPORTS" /YX /FD /c
+# ADD BASE MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x409 /d "NDEBUG"
+# ADD RSC /l 0x409 /d "NDEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386
+# ADD LINK32 php5ts.lib aspell-15.lib kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /dll /machine:I386 /out:"..\..\Release_TS/php_pspell.dll" /libpath:"..\..\Release_TS" /libpath:"..\..\..\aspell\lib"
+
+!ENDIF
+
+# Begin Target
+
+# Name "pspell - Win32 Debug_TS"
+# Name "pspell - Win32 Release_TS"
+# Begin Group "Source Files"
+
+# PROP Default_Filter "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat"
+# Begin Source File
+
+SOURCE=.\pspell.c
+# End Source File
+# End Group
+# Begin Group "Header Files"
+
+# PROP Default_Filter "h;hpp;hxx;hm;inl"
+# Begin Source File
+
+SOURCE=.\php_pspell.h
+# End Source File
+# End Group
+# Begin Group "Resource Files"
+
+# PROP Default_Filter "ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe"
+# End Group
+# End Target
+# End Project
diff --git a/ext/pspell/tests/001.phpt b/ext/pspell/tests/001.phpt
new file mode 100644
index 0000000..0a22352
--- /dev/null
+++ b/ext/pspell/tests/001.phpt
@@ -0,0 +1,116 @@
+--TEST--
+pspell basic tests (warning: may fail with pspell/aspell < GNU Aspell 0.50.3)
+--SKIPIF--
+<?php
+if (!extension_loaded("pspell")) die("skip");
+if (!@pspell_new ("en", "", "", "", (PSPELL_FAST|PSPELL_RUN_TOGETHER))) {
+ die("skip English dictionary is not available");
+}
+?>
+--FILE--
+<?php // $Id$
+
+error_reporting(E_ALL);
+$string = "";
+$string .= "I will not buy this record, it is scratched. ";
+$string .= "Sorry ";
+$string .= "I will not buy this record, it is scratched. ";
+$string .= "Uh, no, no, no. This is a tobacconist's ";
+$string .= "Ah! I will not buy this tobacconist's, it is scratched. ";
+$string .= "No, no, no, no. Tobacco... um... cigarettes (holds up a pack). ";
+$string .= "Ya! See-gar-ets! Ya! Uh... My hovercraft is full of eels. ";
+$string .= "Sorry? ";
+$string .= "My hovercraft (pantomimes puffing a cigarette)... is full of eels (pretends to strike a match). ";
+$string .= "Ahh, matches!";
+
+$pspell = pspell_new ("en", "", "", "", (PSPELL_FAST|PSPELL_RUN_TOGETHER));
+$array = explode(' ',preg_replace('/[^a-zA-Z0-9 ]/','',$string));
+for($i=0,$u=count($array);$i<$u;++$i) {
+ echo $array[$i].' : ';
+ if (!pspell_check($pspell, $array[$i])) {
+ echo "..false\n";
+ echo "Possible spellings: " . join(',',pspell_suggest ($pspell, $array[$i])) . "\n";
+ } else {
+ echo "true\n";
+ }
+}
+?>
+--EXPECTF--
+I : true
+will : true
+not : true
+buy : true
+this : true
+record : true
+it : true
+is : true
+scratched : true
+Sorry : true
+I : true
+will : true
+not : true
+buy : true
+this : true
+record : true
+it : true
+is : true
+scratched : true
+Uh : true
+no : true
+no : true
+no : true
+This : true
+is : true
+a : true
+tobacconists : true
+Ah : true
+I : true
+will : true
+not : true
+buy : true
+this : true
+tobacconists : true
+it : true
+is : true
+scratched : true
+No : true
+no : true
+no : true
+no : true
+Tobacco : true
+um : true
+cigarettes : true
+holds : true
+up : true
+a : true
+pack : true
+Ya : true
+Seegarets : ..false
+Possible spellings:%s,Regrets,%s,Cigarettes,%s
+Ya : true
+Uh : true
+My : true
+hovercraft : true
+is : true
+full : true
+of : true
+eels : true
+Sorry : true
+My : true
+hovercraft : true
+pantomimes : true
+puffing : true
+a : true
+cigarette : true
+is : true
+full : true
+of : true
+eels : true
+pretends : true
+to : true
+strike : true
+a : true
+match : true
+Ahh : ..false
+Possible spellings:%sAh,Aha,%s
+matches : true
diff --git a/ext/pspell/tests/002.phpt b/ext/pspell/tests/002.phpt
new file mode 100644
index 0000000..e6aff66
--- /dev/null
+++ b/ext/pspell/tests/002.phpt
@@ -0,0 +1,42 @@
+--TEST--
+pspell session
+--SKIPIF--
+<?php
+if (!extension_loaded('pspell')) die('skip');
+if (!@pspell_new('en')) die('skip English dictionary is not available');
+?>
+--FILE--
+<?php
+
+$p = pspell_new('en');
+
+var_dump(pspell_check('a'));
+var_dump(pspell_check($p, 'somebogusword'));
+
+var_dump(pspell_add_to_session($p, ''));
+var_dump(pspell_add_to_session($p, 'somebogusword'));
+var_dump(pspell_check($p, 'somebogusword'));
+
+var_dump(pspell_clear_session(new stdclass));
+
+$res = @pspell_clear_session($p);
+if ($res) {
+ var_dump($res);
+ var_dump(pspell_check($p, 'somebogusword'));
+} else {
+ echo "bool(true)\n";
+ echo "bool(false)\n";
+}
+?>
+--EXPECTF--
+Warning: pspell_check() expects exactly 2 parameters, 1 given in %s on line %d
+NULL
+bool(false)
+bool(false)
+bool(true)
+bool(true)
+
+Warning: pspell_clear_session() expects parameter 1 to be long, object given in %s on line %d
+NULL
+bool(true)
+bool(false)
diff --git a/ext/pspell/tests/003.phpt b/ext/pspell/tests/003.phpt
new file mode 100644
index 0000000..4fc34bf
--- /dev/null
+++ b/ext/pspell/tests/003.phpt
@@ -0,0 +1,39 @@
+--TEST--
+pspell_config_ignore
+--SKIPIF--
+<?php
+if (!extension_loaded('pspell')) die('skip');
+if (!@pspell_new('en')) die('skip English dictionary is not available');
+?>
+--FILE--
+<?php
+
+$cfg = pspell_config_create('en', 'british', '', 'iso8859-1');
+$cfg2 = pspell_config_create('en', 'british', '', 'b0rked');
+
+$p = pspell_new_config($cfg);
+var_dump(pspell_check($p, 'yy'));
+
+$p2 = pspell_new_config($cfg2);
+var_dump(pspell_check($p2, 'yy'));
+
+echo "---\n";
+var_dump(pspell_config_ignore($cfg, 2));
+$p = pspell_new_config($cfg);
+var_dump(pspell_check($p, 'yy'));
+
+// segfault it?
+var_dump(pspell_config_ignore($cfg, PHP_INT_MAX));
+
+?>
+--EXPECTF--
+bool(false)
+
+Warning: pspell_new_config(): PSPELL couldn't open the dictionary. reason: The encoding "b0rked" is not known. This could also mean that the file "%sb0rked.%s" could not be opened for reading or does not exist. in %s003.php on line 9
+
+Warning: pspell_check(): 0 is not a PSPELL result index in %s003.php on line 10
+bool(false)
+---
+bool(true)
+bool(true)
+bool(true)
diff --git a/ext/pspell/tests/004.phpt b/ext/pspell/tests/004.phpt
new file mode 100644
index 0000000..80602bc
--- /dev/null
+++ b/ext/pspell/tests/004.phpt
@@ -0,0 +1,35 @@
+--TEST--
+pspell configs
+--SKIPIF--
+<?php
+if (!extension_loaded('pspell')) die('skip');
+if (!@pspell_new('en')) die('skip English dictionary is not available');
+?>
+--FILE--
+<?php
+
+$cfg = pspell_config_create('en', 'british', '', 'iso8859-1');
+var_dump(pspell_config_mode($cfg, PSPELL_BAD_SPELLERS));
+
+var_dump(pspell_config_runtogether($cfg, false));
+$p = pspell_new_config($cfg);
+var_dump(pspell_check($p, 'theoasis'));
+
+echo "---\n";
+
+// now it should pass
+var_dump(pspell_config_runtogether($cfg, true));
+$p = pspell_new_config($cfg);
+var_dump(pspell_check($p, 'theoasis'));
+
+var_dump(pspell_config_runtogether($cfg, NULL))
+
+?>
+--EXPECT--
+bool(true)
+bool(true)
+bool(false)
+---
+bool(true)
+bool(true)
+bool(true)
diff --git a/ext/pspell/tests/005.phpt b/ext/pspell/tests/005.phpt
new file mode 100644
index 0000000..17ed190
--- /dev/null
+++ b/ext/pspell/tests/005.phpt
@@ -0,0 +1,46 @@
+--TEST--
+pspell configs
+--SKIPIF--
+<?php
+if (!extension_loaded('pspell')) die('skip');
+if (!@pspell_new('en')) die('skip English dictionary is not available');
+?>
+--FILE--
+<?php
+
+$wordlist = dirname(__FILE__).'/wordlist.txt';
+
+var_dump(pspell_new_personal(__FILE__, 'en'));
+$p = pspell_new_personal($wordlist, 'en');
+
+var_dump(pspell_check($p, 'dfnvnsafksfksf'));
+
+echo "--\n";
+$cfg = pspell_config_create('en');
+var_dump(pspell_config_personal($cfg, "$wordlist.tmp"));
+$p = pspell_new_config($cfg);
+
+copy($wordlist, "$wordlist.tmp");
+
+var_dump(pspell_check($p, 'ola'));
+var_dump(pspell_add_to_personal($p, 'ola'));
+var_dump(pspell_check($p, 'ola'));
+
+echo "--\n";
+var_dump(pspell_save_wordlist($p));
+var_dump(strpos(file_get_contents("$wordlist.tmp"), 'ola') !== FALSE);
+
+unlink("$wordlist.tmp");
+?>
+--EXPECTF--
+Warning: pspell_new_personal(): PSPELL couldn't open the dictionary. reason: The file "%s005.php" is not in the proper format. in %s005.php on line 5
+bool(false)
+bool(true)
+--
+bool(true)
+bool(false)
+bool(true)
+bool(true)
+--
+bool(true)
+bool(true)
diff --git a/ext/pspell/tests/wordlist.txt b/ext/pspell/tests/wordlist.txt
new file mode 100644
index 0000000..e9d8da7
--- /dev/null
+++ b/ext/pspell/tests/wordlist.txt
@@ -0,0 +1,5 @@
+personal_ws-1.1 en 4
+dfnvnsafksfksf
+fg
+iufrsn
+jsksjfsjf