Initial commit
authorMyhailo Danylenko <isbear@ukrpost.net>
Tue, 10 Nov 2009 18:27:16 +0200
changeset 0 7707b26e82fd
child 1 320e4393785a
Initial commit
.gitignore
CMakeLists.txt
COPYING
config.h
help/en/hlp_eval.txt
help/en/hlp_if.txt
help/en/hlp_let.txt
help/en/hlp_multi.txt
yaubil.c
yaubil.rc
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.gitignore	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,25 @@
+# ignore generated by libtool files (no more necessary)
+*.a
+*.o
+*.lo
+*.la
+*.so
+*.lai
+*.so.*
+# ignore backup files
+*~
+# ignore docs: they are not important and apiref is generated on the fly
+*.html
+# ignore generated makefile
+Makefile
+# ignore cmake files
+*.cmake
+CMake*
+CPack*
+# ignore generated header
+config.h
+# ignore resulting bundles
+*.deb
+*.tar.bz2
+# ignore build dir
+build
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/CMakeLists.txt	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,69 @@
+## Copyright 2009 Myhailo Danylenko
+# This file is part of mcabber module writing howto examples.
+#
+# Examples are 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 2 of the License, 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 this program.  If not, see <http://www.gnu.org/licenses/>.
+
+cmake_minimum_required(VERSION 2.6) 
+project(yaubil C) 
+
+## Target definitions
+add_library(yaubil MODULE yaubil.c) 
+
+## User settable options
+set(MCABBER_INCLUDE_DIR "${yaubil_SOURCE_DIR}/../include" 
+    CACHE FILEPATH "Path to mcabber headers") 
+ 
+## Packaging information
+set(CPACK_PACKAGE_NAME libmcabber-yaubil)
+set(CPACK_PACKAGE_VERSION "0.0.1")
+set(CPACK_PACKAGE_VENDOR "IsBear")
+set(CPACK_PACKAGE_CONTACT "Myhailo Danylenko <isbear@ukrpost.net>")
+set(CPACK_PACKAGE_DESCRIPTION_SUMMARY "Mcabber example modularized yaubil implementation")
+set(CPACK_RESOURCE_FILE_LICENSE ${yaubil_SOURCE_DIR}/COPYING)
+set(CPACK_SOURCE_GENERATOR TBZ2)
+set(CPACK_GENERATOR DEB CACHE TEXT "Binary package generator, eg DEB, RPM, TGZ, NSIS...")
+set(CPACK_DEBIAN_PACKAGE_SECTION libs)
+find_program(DPKG_EXECUTABLE dpkg)
+if(DPKG_EXECUTABLE)
+	execute_process(COMMAND ${DPKG_EXECUTABLE} --print-architecture OUTPUT_VARIABLE CPACK_DEBIAN_PACKAGE_ARCHITECTURE OUTPUT_STRIP_TRAILING_WHITESPACE)
+else()
+	set(CPACK_DEBIAN_PACKAGE_ARCHITECTURE i386 CACHE TEXT "Architecture of package")
+endif()
+set(CPACK_PACKAGE_FILE_NAME "${CPACK_PACKAGE_NAME}_${CPACK_PACKAGE_VERSION}_${CPACK_DEBIAN_PACKAGE_ARCHITECTURE}")
+set(CPACK_SOURCE_IGNORE_FILES "/\\\\..*;\\\\.swp;~$;/build/;\\\\.tar\\\\.;\\\\.deb;\\\\.so")
+include(CPack)
+
+## Check for build dependencies
+find_package(PkgConfig REQUIRED) 
+pkg_check_modules(GLIB REQUIRED glib-2.0) 
+pkg_check_modules(GMODULE REQUIRED gmodule-2.0) 
+#pkg_check_modules(LM REQUIRED loudmouth-1.0) 
+
+## Compiler setup
+include_directories(SYSTEM ${GLIB_INCLUDE_DIRS} 
+                    ${GMODULE_INCLUDE_DIRS} 
+                    ${LM_INCLUDE_DIRS}) 
+target_link_libraries(yaubil ${GLIB_LIBRARIES} 
+                      ${GMODULE_LIBRARIES}
+					  ${LM_LIBRARIES})
+include_directories(${yaubil_SOURCE_DIR} 
+                    ${yaubil_BINARY_DIR} 
+                    ${MCABBER_INCLUDE_DIR}) 
+
+## Installation
+install(TARGETS yaubil DESTINATION lib/mcabber) 
+install(FILES COPYING yaubil.rc DESTINATION share/doc/${CPACK_PACKAGE_NAME})
+install(DIRECTORY help DESTINATION share/mcabber)
+
+## The End ## vim: se ts=4: ##
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/COPYING	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,339 @@
+		    GNU GENERAL PUBLIC LICENSE
+		       Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+			    Preamble
+
+  The licenses for most software are designed to take away your
+freedom to share and change it.  By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users.  This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it.  (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.)  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+  To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have.  You must make sure that they, too, receive or can get the
+source code.  And you must show them these terms so they know their
+rights.
+
+  We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+  Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software.  If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+  Finally, any free program is threatened constantly by software
+patents.  We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary.  To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+		    GNU GENERAL PUBLIC LICENSE
+   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+  0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License.  The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language.  (Hereinafter, translation is included without limitation in
+the term "modification".)  Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope.  The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+  1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+  2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+    a) You must cause the modified files to carry prominent notices
+    stating that you changed the files and the date of any change.
+
+    b) You must cause any work that you distribute or publish, that in
+    whole or in part contains or is derived from the Program or any
+    part thereof, to be licensed as a whole at no charge to all third
+    parties under the terms of this License.
+
+    c) If the modified program normally reads commands interactively
+    when run, you must cause it, when started running for such
+    interactive use in the most ordinary way, to print or display an
+    announcement including an appropriate copyright notice and a
+    notice that there is no warranty (or else, saying that you provide
+    a warranty) and that users may redistribute the program under
+    these conditions, and telling the user how to view a copy of this
+    License.  (Exception: if the Program itself is interactive but
+    does not normally print such an announcement, your work based on
+    the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole.  If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works.  But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+  3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+    a) Accompany it with the complete corresponding machine-readable
+    source code, which must be distributed under the terms of Sections
+    1 and 2 above on a medium customarily used for software interchange; or,
+
+    b) Accompany it with a written offer, valid for at least three
+    years, to give any third party, for a charge no more than your
+    cost of physically performing source distribution, a complete
+    machine-readable copy of the corresponding source code, to be
+    distributed under the terms of Sections 1 and 2 above on a medium
+    customarily used for software interchange; or,
+
+    c) Accompany it with the information you received as to the offer
+    to distribute corresponding source code.  (This alternative is
+    allowed only for noncommercial distribution and only if you
+    received the program in object code or executable form with such
+    an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it.  For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable.  However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+  4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License.  Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+  5. You are not required to accept this License, since you have not
+signed it.  However, nothing else grants you permission to modify or
+distribute the Program or its derivative works.  These actions are
+prohibited by law if you do not accept this License.  Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+  6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions.  You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+  7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all.  For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices.  Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+  8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded.  In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+  9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number.  If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation.  If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+  10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission.  For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this.  Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+			    NO WARRANTY
+
+  11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU.  SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+  12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+		     END OF TERMS AND CONDITIONS
+
+	    How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    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 2 of the License, 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 this program; if not, write to the Free Software Foundation, Inc.,
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+    Gnomovision version 69, Copyright (C) year name of author
+    Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary.  Here is a sample; alter the names:
+
+  Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+  `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+  <signature of Ty Coon>, 1 April 1989
+  Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs.  If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/config.h	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,8 @@
+
+#ifndef LOCAL_CONFIG_H
+#define LOCAL_CONFIG_H
+
+#define MODULES_ENABLE 1
+
+# endif
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/help/en/hlp_eval.txt	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,28 @@
+
+ /EVAL [expression]
+
+Evaluates expression and executes result string as command.
+Expression evaluation rules:
+* all operators have the same precedence;
+* parenthesis searching will not take into account strings;
+* supported operators:
+  . string concatenation
+  + binary plus / string concatenation
+  - binary minus
+  * multiplication / string replication
+  / division
+  % division remain
+  < binary / string less than
+  > binary / string greater than
+  = binary / string equality
+* strings must be quoted with double quotes ("..."), within string backslash is used for character escaping (\\ and \" are only useful applications);
+* parentheses can be used to change operators precedence, nested parentheses are supported as long as they are balanced;
+* numbers can be specified as plain decimal numbers, minus in front of digit will be recognized as a part of number;
+* mcabber variables can be specified by name, though only subset of possible names is suppoted:
+  - variable must start from latin alphabetic symbol;
+  - inside of variable name also are allowed digits, - and _.
+
+Example:
+ /eval "echo 2 + 2 = " . (2+2)
+ /eval "set ping_interval = " . (ping_interval + 1)
+Note: last example probably should use /let command.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/help/en/hlp_if.txt	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,12 @@
+
+ /IF [expression]
+ /THEN command
+ /ELSE command
+
+Evaluates expression and stores true if expression evaluates to non-zero integer or non-unset string, false if it evaluates to unset value, integer zero or unset string.
+Then and else commands can be called any number of times - they will use the same stored value.
+For details on expression syntax see help for eval command.
+Example:
+ /if fifo_name
+ /then load fifo
+ /else echo "fifo_name not set, fifo module not loaded"
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/help/en/hlp_let.txt	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,7 @@
+
+ /LET variable = [expression]
+
+Evaluates expression and assigns result to mcabber variable.
+For details on expression syntax, see help for eval command.
+Example:
+ /let ping_interval = ping_interval + 1
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/help/en/hlp_multi.txt	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,7 @@
+
+ /MULTI command[;command[...]]
+
+Runs specified mcabber commands one by one. Note, that it will not honour any backslash-escapes, parentheses, quotes etc - it will just split line on semicolons.
+Example:
+ /set hook-post-connect = multi group fold Transports; group fold Old
+ /multi if fifo_name; then load fifo; else echo "fifo_name not set, fifo module not loaded"
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/yaubil.c	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,852 @@
+/*
+ * yaubil.c             -- Yet Another Useless Built-In Language
+ *
+ * Copyrigth (C) 2009      Myhailo Danylenko <isbear@ukrpost.net>
+ *
+ * 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 2 of the License, 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 this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
+ * USA
+ */
+
+#include <glib.h>
+#include <gmodule.h>
+#include <string.h>
+
+#include "commands.h"
+#include "compl.h"
+#include "logprint.h"
+#include "settings.h"
+
+static gboolean ifresult = TRUE;
+
+#define MSGPREFIX "yaubil: "
+
+#define TYPE_UNDEF ( 0 )
+#define TYPE_STR   ( 1 )
+#define TYPE_INT   ( 2 )
+
+#define STATE_LVALUE ( 1 )
+#define STATE_OP     ( 2 )
+#define STATE_RVALUE ( 3 )
+
+typedef struct {
+	int   type;
+	int   int_value;
+	char *str_value;
+} value_t;
+
+typedef struct {
+	char op;
+	gboolean (*handler) (value_t *l, value_t *r);
+} op_t;
+
+static int check_value_type (const char *value)
+{
+	if (value) {
+		const char *e;
+		gboolean    integer = (*value == '-') ? FALSE : (g_ascii_isdigit (*value) ? TRUE : FALSE);
+
+		if (integer) {
+			for (e = value + 1; *e; ++e) {
+				if (g_ascii_isdigit (*e))
+					integer = TRUE;
+				else {
+					integer = FALSE;
+					break;
+				}
+			}
+		}
+
+		if (integer)
+			return TYPE_INT;
+		else
+			return TYPE_STR;
+	} else
+		return TYPE_UNDEF;
+}
+
+static gboolean op_concat (value_t *l, value_t *r)
+{
+	GString *res = g_string_new (NULL);;
+
+	if (l->type == TYPE_INT)
+		g_string_append_printf (res, "%d", l->int_value);
+	else if (l->type == TYPE_STR)
+		g_string_append (res, l->str_value);
+
+	if (r->type == TYPE_INT)
+		g_string_append_printf (res, "%d", r->int_value);
+	else if (r->type == TYPE_STR)
+		g_string_append (res, r->str_value);
+	
+	l->type = TYPE_STR;
+	g_free (l->str_value);
+	l->str_value = g_string_free (res, FALSE);
+
+	return TRUE;
+}
+
+static gboolean op_plus (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		// not free value here
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		// not free value here
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		// integer
+		l->int_value += r->int_value;
+		g_free (l->str_value);
+		l->str_value = NULL;
+		g_free (r->str_value);
+		r->str_value = NULL;
+	} else {
+		// convert both to strings
+		if (l->type == TYPE_INT) {
+			l->type = TYPE_STR;
+			if (!l->str_value)
+				l->str_value = g_strdup_printf ("%d", l->int_value);
+		}
+		if (r->type == TYPE_INT) {
+			r->type = TYPE_STR;
+			if (!r->str_value)
+				r->str_value = g_strdup_printf ("%d", r->int_value);
+		}
+
+		{
+			char *tmp = l->str_value;
+			l->str_value = g_strdup_printf ("%s%s", tmp, r->str_value);
+			g_free (tmp);
+		}
+	}
+
+	return TRUE;
+}
+
+static gboolean op_minus (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		g_free (l->str_value);
+		l->str_value = NULL;
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		g_free (r->str_value);
+		r->str_value = NULL;
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT)
+		l->int_value -= r->int_value;
+	else
+		return FALSE;
+
+	return TRUE;
+}
+
+static gboolean op_multiply (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		// not free value here
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		g_free (r->str_value);
+		r->str_value = NULL;
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		l->int_value *= r->int_value;
+		g_free (l->str_value);
+		l->str_value = NULL;
+	} else if (r->type == TYPE_INT) {
+		GString *res = g_string_new (NULL);
+		int      i;
+		for (i = r->int_value; i; --i)
+			g_string_append (res, l->str_value);
+		g_free (l->str_value);
+		l->str_value = g_string_free (res, FALSE);
+	} else
+		return FALSE;
+
+	return TRUE;
+}
+
+static gboolean op_divide (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		g_free (l->str_value);
+		l->str_value = NULL;
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		g_free (r->str_value);
+		r->str_value = NULL;
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		if (r->int_value == 0) {
+			scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "/: Error: division by zero.");
+			return FALSE;
+		}
+		l->int_value /= r->int_value;
+	} else
+		return FALSE;
+
+	return TRUE;
+}
+
+static gboolean op_remain (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		g_free (l->str_value);
+		l->str_value = NULL;
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		g_free (r->str_value);
+		r->str_value = NULL;
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		if (r->int_value == 0) {
+			scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "%%: Error: division by zero.");
+			return FALSE;
+		}
+		l->int_value %= r->int_value;
+	} else
+		return FALSE;
+
+	return TRUE;
+}
+
+static gboolean op_equal (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		// not free value here
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		// not free value here
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		// integer
+		l->int_value = (l->int_value == r->int_value) ? 1 : 0;
+		g_free (l->str_value);
+		l->str_value = NULL;
+		g_free (r->str_value);
+		r->str_value = NULL;
+	} else {
+		// convert both to strings
+		if (l->type == TYPE_INT) {
+			l->type = TYPE_STR;
+			if (!l->str_value)
+				l->str_value = g_strdup_printf ("%d", l->int_value);
+		}
+		if (r->type == TYPE_INT) {
+			r->type = TYPE_STR;
+			if (!r->str_value)
+				r->str_value = g_strdup_printf ("%d", r->int_value);
+		}
+
+		{
+			char *tmp = l->str_value;
+			l->type      = TYPE_INT;
+			l->int_value = (g_strcmp0 (l->str_value, r->str_value) == 0) ? 1 : 0;
+			g_free (tmp);
+		}
+	}
+
+	return TRUE;
+}
+
+static gboolean op_lt (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		// not free value here
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		// not free value here
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		// integer
+		l->int_value = (l->int_value < r->int_value) ? 1 : 0;
+		g_free (l->str_value);
+		l->str_value = NULL;
+		g_free (r->str_value);
+		r->str_value = NULL;
+	} else {
+		// convert both to strings
+		if (l->type == TYPE_INT) {
+			l->type = TYPE_STR;
+			if (!l->str_value)
+				l->str_value = g_strdup_printf ("%d", l->int_value);
+		}
+		if (r->type == TYPE_INT) {
+			r->type = TYPE_STR;
+			if (!r->str_value)
+				r->str_value = g_strdup_printf ("%d", r->int_value);
+		}
+
+		{
+			char *tmp = l->str_value;
+			l->type      = TYPE_INT;
+			l->int_value = (g_strcmp0 (l->str_value, r->str_value) < 0) ? 1 : 0;
+			g_free (tmp);
+		}
+	}
+
+	return TRUE;
+}
+
+static gboolean op_gt (value_t *l, value_t *r)
+{
+	if (l->type == TYPE_UNDEF) {
+		l->type      = TYPE_INT;
+		l->int_value = 0;
+	}
+	if (r->type == TYPE_UNDEF) {
+		r->type      = TYPE_INT;
+		r->int_value = 0;
+	}
+	
+	if (l->type == TYPE_STR && check_value_type (l->str_value) == TYPE_INT) {
+		l->type      = TYPE_INT;
+		l->int_value = atoi (l->str_value);
+		// not free value here
+	}
+	if (r->type == TYPE_STR && check_value_type (r->str_value) == TYPE_INT) {
+		r->type      = TYPE_INT;
+		r->int_value = atoi (r->str_value);
+		// not free value here
+	}
+
+	if (l->type == TYPE_INT && r->type == TYPE_INT) {
+		// integer
+		l->int_value = (l->int_value > r->int_value) ? 1 : 0;
+		g_free (l->str_value);
+		l->str_value = NULL;
+		g_free (r->str_value);
+		r->str_value = NULL;
+	} else {
+		// convert both to strings
+		if (l->type == TYPE_INT) {
+			l->type = TYPE_STR;
+			if (!l->str_value)
+				l->str_value = g_strdup_printf ("%d", l->int_value);
+		}
+		if (r->type == TYPE_INT) {
+			r->type = TYPE_STR;
+			if (!r->str_value)
+				r->str_value = g_strdup_printf ("%d", r->int_value);
+		}
+
+		{
+			char *tmp = l->str_value;
+			l->type      = TYPE_INT;
+			l->int_value = (g_strcmp0 (l->str_value, r->str_value) > 0) ? 1 : 0;
+			g_free (tmp);
+		}
+	}
+
+	return TRUE;
+}
+
+static op_t operators[] = {
+	{ '.', op_concat   },
+	{ '+', op_plus     },
+	{ '-', op_minus    },
+	{ '*', op_multiply },
+	{ '/', op_divide   },
+	{ '%', op_remain   },
+	{ '=', op_equal    },
+	{ '<', op_lt       },
+	{ '>', op_gt       },
+	{ 0,   NULL        },
+};
+
+static void destroy_value (value_t *value)
+{
+	if (value->str_value)
+		g_free (value->str_value);
+	g_free (value);
+}
+
+static value_t *process_expression (const char *str, gsize len)
+{
+	const char *strend = str + len;
+	const char *p;
+	op_t       *op     = NULL;
+	int         state  = STATE_LVALUE;
+	value_t     val    = {
+		.type      = TYPE_UNDEF,
+		.int_value = 0,
+		.str_value = NULL,
+	};
+
+	for (p = str; *p && p < strend; ++p) {
+		switch (state) {
+		case STATE_LVALUE:
+		case STATE_RVALUE:
+
+			if (*p == ' ')
+				break;
+
+			{
+				value_t rval = {
+					.type      = TYPE_UNDEF,
+					.int_value = 0,
+					.str_value = NULL,
+				};
+
+				if (g_ascii_isdigit (*p) || *p == '-') { // integer // XXX: no unary operators for now...
+
+					const char *e;
+
+					rval.type = TYPE_INT;
+
+					for (e = p + 1; g_ascii_isdigit (*e) && e < strend; ++e); // TODO: does atoi handle 0x etc? then also allow this?
+					
+					{
+						char *v = g_strndup (p, e - p);
+						rval.int_value = atoi (v);
+						g_free (v);
+					}
+
+					p = e - 1;
+
+				} else if (*p == '"') { // string
+
+					const char *e;
+
+					rval.type = TYPE_STR;
+
+					{
+						gboolean  finished = FALSE;
+						gboolean  escape   = FALSE;
+						GString  *v        = g_string_new (NULL);
+
+						for (e = p + 1; *e && e < strend; ++e) {
+							switch (*e) {
+							case '\\':
+								if (!escape)
+									escape = TRUE;
+								else
+									escape = FALSE;
+								break;
+							case '"':
+								if (!escape) {
+									finished = TRUE;
+									break;
+								} else
+									escape = FALSE;
+								break;
+							default:
+								escape = FALSE;
+								break;
+							}
+
+							if (finished)
+								break;
+
+							if (!escape)
+								g_string_append_c (v, *e);
+						};
+
+						if (!finished) {
+							scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: Unmatched quote.");
+							g_string_free (v, TRUE);
+							g_free (val.str_value);
+							return NULL;
+						}
+
+						rval.str_value = g_string_free (v, FALSE);
+					}
+
+					p = e;
+
+				} else if (g_ascii_isalpha (*p)) { // variable (MUST start from alpha)
+
+					const char *e;
+
+					for (e = p + 1; (g_ascii_isalnum (*e) || *e == '-' || *e == '_') && e < strend; ++e);
+
+					char       *name = g_strndup (p, e - p);
+					const char *value = settings_opt_get (name);
+					g_free (name);
+
+					rval.type = check_value_type (value);
+					if (rval.type == TYPE_INT)
+						rval.int_value = atoi (value);
+					else if (rval.type == TYPE_STR)
+						rval.str_value = g_strdup (value);
+
+					p = e - 1;
+
+				} else if (*p == '(') {
+					
+					const char *e;
+
+					{
+						int      l        = 0;
+						gboolean finished = FALSE;
+						
+						// XXX maybe we should just pass end of line and let it return real length...
+						for (e = p + 1; e < strend; ++e) {
+							if (*e == '(')
+								++l;
+							else if (*e == ')') {
+								--l;
+								if (l < 0) {
+									finished = TRUE;
+									break;
+								}
+							}
+						}
+
+						if (!finished) {
+							scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: Unmatched parenthesis.");
+							g_free (val.str_value);
+							return NULL;
+						}
+					}
+
+					{
+						value_t *n = process_expression (p + 1, e - p - 1);
+						
+						if (!n) {
+							scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: Error in subexpression.");
+							g_free (val.str_value);
+							return NULL;
+						}
+
+						rval.type      = n->type;
+						rval.int_value = n->int_value;
+						rval.str_value = g_strdup (n->str_value);
+						destroy_value (n);
+					}
+
+					p = e;
+
+				} else {
+					scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: unrecognized symbols.");
+					g_free (val.str_value);
+					return NULL;
+				}
+
+				if (state == STATE_RVALUE) {
+					if (op->handler) {
+						if (!op->handler (&val, &rval)) {
+							scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: operand argument types mismatch.");
+							g_free (val.str_value);
+							g_free (rval.str_value);
+							return NULL;
+						}
+					}
+					g_free (rval.str_value);
+				} else {
+					val.type      = rval.type;
+					val.int_value = rval.int_value;
+					val.str_value = rval.str_value;
+				}
+				
+				state = STATE_OP;
+			}
+
+			break;
+
+		case STATE_OP:
+
+			if (*p == ' ')
+				break;
+
+			{
+				op_t     *operator;
+				gboolean  found    = FALSE;
+
+				for (operator = operators; operator->op; ++operator) {
+					if (operator->op == *p) {
+						op    = operator;
+						found = TRUE;
+						break;
+					}
+				}
+
+				if (!found) {
+					scr_LogPrint (LPRINT_NORMAL, MSGPREFIX "Error: unknown operator.");
+					g_free (val.str_value);
+				}
+			}
+
+			state = STATE_RVALUE;
+
+			break;
+		default:
+			break;
+		}
+	}
+
+	{ // return value
+		value_t *rval = g_new (value_t, 1);
+
+		rval->type      = val.type;
+		rval->int_value = val.int_value;
+		rval->str_value = val.str_value;
+
+		return rval;
+	}
+}
+
+static void do_eval (char *arg)
+{
+	value_t *val = process_expression (arg, strlen (arg));
+
+	if (!val) {
+		scr_LogPrint (LPRINT_NORMAL, "eval: Evaluation error.");
+		return;
+	}
+
+	if (val->type == TYPE_STR)
+		process_command (val->str_value, TRUE);
+	else
+		scr_LogPrint (LPRINT_NORMAL, "eval: Expression does not result in string.");
+
+	destroy_value (val);
+}
+
+static void do_let (char *arg)
+{
+	char     *varname;
+	value_t  *value;
+	int       namelen;
+	char     *val     = strchr (arg, '=');
+
+	if (!val) {
+		scr_LogPrint (LPRINT_NORMAL, "let: Syntax error: no equal sign in line.");
+		return;
+	}
+	
+	{
+		char *p = val;
+
+		for (p = val - 1; p >= arg && *p == ' '; --p);
+		if (p < arg) {
+			scr_LogPrint (LPRINT_NORMAL, "let: Syntax error: no destination variable name specified.");
+			return;
+		}
+
+		namelen = p + 1 - arg;
+	}
+
+	// evaluate expression
+	value = process_expression (val + 1, strlen (val + 1));
+
+	if (!value) {
+		scr_LogPrint (LPRINT_NORMAL, "let: Evaluation error.");
+		return;
+	}
+
+	if (value->type == TYPE_INT) {
+		value->type      = TYPE_STR;
+		value->str_value = g_strdup_printf ("%d", value->int_value);
+	}
+
+	{ // assign value
+		char *varname = g_strndup (arg, namelen);
+
+		if (value->str_value)
+			settings_set (SETTINGS_TYPE_OPTION, varname, value->str_value);
+		else
+			settings_del (SETTINGS_TYPE_OPTION, varname);
+
+		g_free (varname);
+	}
+
+	destroy_value (value);
+}
+
+static void do_if (char *arg)
+{
+	value_t *val = process_expression (arg, strlen (arg));
+
+	if (!val) {
+		scr_LogPrint (LPRINT_NORMAL, "if: Evaluation error.");
+		return;
+	}
+
+	if (val->type == TYPE_UNDEF) {
+		val->type = TYPE_INT;
+		val->int_value = 0;
+	}
+	if (val->type == TYPE_STR && check_value_type (val->str_value) == TYPE_INT) {
+		val->type = TYPE_INT;
+		val->int_value = atoi (val->str_value);
+		g_free (val->str_value);
+		val->str_value = NULL;
+	}
+
+	if (val->type == TYPE_INT)
+		ifresult = val->int_value ? TRUE : FALSE;
+	else if (val->str_value)
+		ifresult = TRUE;
+	else
+		ifresult = FALSE;
+
+	destroy_value (val);
+}
+
+static void do_then (char *arg)
+{
+	if (ifresult)
+		process_command (arg, TRUE);
+}
+
+static void do_else (char *arg)
+{
+	if (!ifresult)
+		process_command (arg, TRUE);
+}
+
+static void do_multi (char *arg)
+{
+	char *end;
+	char *start = arg;
+	
+	for (end = strchr (start, ';'); end; end = strchr (start, ';')) {
+
+		// execute command
+		char *command = g_strndup (start, end - start);
+		process_command (command, TRUE);
+		g_free (command);
+
+		// skip leading spaces
+		for (start = end + 1; *start == ' '; ++start);
+	}
+
+	if (*start)
+		process_command (start, TRUE);
+}
+
+const gchar *g_module_check_init (GModule *module)
+{
+	cmd_add ("multi", "", COMPL_CMD, COMPL_CMD, do_multi, NULL);
+	cmd_add ("if", "", 0, 0, do_if, NULL);
+	cmd_add ("then", "", COMPL_CMD, COMPL_CMD, do_then, NULL);
+	cmd_add ("else", "", COMPL_CMD, COMPL_CMD, do_else, NULL);
+	cmd_add ("eval", "", 0, 0, do_eval, NULL);
+	cmd_add ("let", "", 0, 0, do_let, NULL);
+
+	return NULL;
+}
+
+void g_module_unload (GModule *module)
+{
+	cmd_del ("multi");
+	cmd_del ("if");
+	cmd_del ("then");
+	cmd_del ("else");
+	cmd_del ("eval");
+	cmd_del ("let");
+}
+
+/* The End */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/yaubil.rc	Tue Nov 10 18:27:16 2009 +0200
@@ -0,0 +1,3 @@
+
+load yaubil
+