Version 1.0.3
revised 2009-04-17 by Rex Gozar
You can connect a web server to a Universe database via u2pipe. Windows platforms use wininetd as the socket listener. Unix/Linux platforms can use xinetd or inetd. This document explains how to do it.
There are better, more sophisticated ways of doing this. You could use RedBack (or whatever its new name is), UniObjects, mvNet, or any other commercial solution. But I'm tired of the "barriers to entry" in the PICK programming world, and I wanted to provide a simple open source alternative for connecting web servers to IBM U2 databases.
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.
u2pipe 6789/tcp # u2pipe Web Service
u2pipe stream tcp nowait root /usr/local/bin/u2pipe
service u2pipe
{
instances = 3
socket_type = stream
wait = no
user = root
server = /usr/local/bin/u2pipe
nice = 10
disable = no
only_from = localhost 10.0.0.0 192.168.1.132
}
server_args = /etc/sales_account.ini
You typically don't have to change wininetd.conf, but if you want to the format is:
port user:password full-u2pipe-path [full-inifile-path]
In Windows Services, you must change wininetd properties from "Manual" to "Automatic" for the service to start at reboot.
You must restart the wininetd service for changes to wininetd.conf to take effect. This is easy to do from the DOS command line by typing:
net stop wininetd
net start wininetd
Changes to u2pipe.ini take effect immediately (does not require restart).
You can create different ini files to log into different machines or accounts, and tie them to specific port numbers in wininetd.conf (or inetd.conf, or xinetd.d files -- you get the idea).
Sample u2pipe.ini with minimum configuration:
[U2PIPE] hostname=localhost user=bgates password=iluvlinux! account=SALES
Run both of these tests to make sure that u2pipe and wininetd are configured correctly. Error messages show up in the LOGFILE (specified in the ini file) or the Windows Event Viewer.
u2pipe is designed to take a request from stdin, process it, and send the response to stdout.
REQUEST format:
where
u2pipe reads in the request, strips off the six-digit request length, and prepends the optional RUN parameter (from the ini file above).
u2pipe uses InterCall functions ic_opensession(), ic_execute(), and ic_quit() to process the command.
u2pipe returns the length of the output and the output itself in the format of a response.
RESPONSE format:
where
So you got the "DATE" command to work – Yahoo! break out the champagne (or maybe some of that "free beer")! No, wait a minute – I suppose you want to run something a little more complicated, huh?
I'm going to outline a simple implementation. You'll need to add your own features, like error return handling, beefier encryption, etc. The purpose of u2pipe is to provide a connection between a webserver and Universe; how to use that connection is up to you.
1. We'll devise our own data format for exchanging information between the web server and Universe.
2. We'll write a function on the frontend.
3. We'll write a Universe BASIC program on the backend
4. We'll set the RUN parameter in u2pipe.ini to run our backend program.
5. We'll test it.
1. Devising a data exchange format
This is a simple implementation, so I've decided that the webserver should be able to send TCL command strings with arguments and get the output in return. With that in mind, I've chosen to simply encode my command string and the output returned using base64. I've also decided to wrap the encoded output in parentheses "()" to help me find it in the response.
If I wanted to get fancy, I could encapsulate my request in XML (maybe like SOAP) and then encode, encrypt, or otherwise mangle the request before sending it to the backend. Conversely, I could also require the response to be XML formatted and encoded. Feel free to make it as simple or complicated as you wish.
2. Writing a frontend function.
ASP, ASP.NET, JSP, PHP, ColdFusion... it does not matter what frontend you use. As long as your frontend can either (a) communicate on a socket, or (b) retrieve the contents of a web page, you can write a function for interacting with u2pipe.
2a. Writing a frontend function: ColdFusion
I've already written a function for you. Change it as you need to implement your data exchange format.
<cffunction name="decryptString" output="false" returntype="any">
<cfargument name="text" type="string" required="true">
<!--- *TODO* substitute your own encryption --->
<cfreturn toString(toBinary(text))>
</cffunction>
<cffunction name="encryptString" output="false" returntype="string">
<cfargument name="text" type="any" required="true">
<!--- *TODO* substitute your own encryption --->
<cfreturn toBase64(text, "us-ascii")>
</cffunction>
<cffunction name="serviceRequest" output="false" returntype="string">
<cfargument name="myCommand" type="string" required="true">
<!--- some constants --->
<cfset var HEADERSZ = 6>
<cfset var HOSTURL = "http://yourhost:6789">
<!--- local variables --->
<cfset var myLength = "">
<cfset var myRequest = "">
<cfset var myResponse = "">
<cfset var myOutput = "">
<cfset var bptr = 0>
<cfset var eptr = 0>
<!--- Encrypt the command --->
<cfset myCommand = encryptString(myCommand)>
<!--- Build the request from the command --->
<cfset myLength = right(repeatString("0", HEADERSZ) & len(myCommand), HEADERSZ)>
<cfset myRequest = HOSTURL & "/" & myLength & myCommand>
<cftry>
<cfhttp url="#myRequest#" timeout="60" />
<!--- Parse the output from the response --->
<cfset myResponse = cfhttp.fileContent>
<cfset myLength = mid(myResponse, 1, HEADERSZ)>
<cfset myOutput = mid(myResponse, (HEADERSZ + 1), myLength)>
<!--- Parse output markers --->
<!--- We've decided that output will be formatted: !(base64text) --->
<cfset bptr = find("(", myOutput)>
<cfset eptr = find(")", myOutput)>
<cfif bptr lt eptr>
<cfset myOutput = mid(myOutput, (bptr+1), (eptr-bptr-1))>
<cfelse>
<cfthrow message="missing or invalid markers!">
</cfif>
<!--- Decrypt the output --->
<cfset myOutput = decryptString(myOutput)>
<cfcatch type="any">
<!--- *TODO* you'll need some exception handling --->
</cfcatch>
</cftry>
<cfreturn myOutput>
</cffunction>
<!--- now get some data from Universe --->
<cfset command = "LIST VOC SAMPLE 3">
<cfset output = serviceRequest(command)>
<cfoutput>
<pre>
#output#
</pre>
</cfoutput>
2b. Writing a frontend function: PHP
I've written a function for you PHP'ers too. Again, change it as you need to implement your data exchange format.
<?php
function decryptString($text) {
// *TODO* substitute your own encryption
return base64_decode($text);
}
function encryptString($text) {
// *TODO* substitute your own encryption
return base64_encode($text);
}
function serviceRequest($myCommand) {
// set constants
define("HEADERSZ", 6);
define("HOSTIP", "yourhost");
define("HOSTPORT", 6789);
if (($socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) < 0 ) {
// *TODO* need to handle error"
die("Could not create socket!\n");
exit(1);
}
socket_set_option($socket, SOL_SOCKET, SO_RCVTIMEO, array('sec'=>20, 'usec'=>0));
if (!socket_connect($socket, HOSTIP, HOSTPORT)) {
// *TODO* need to handle error"
die("Unable to connect!\n");
exit(1);
}
$myCommand = encryptString($myCommand);
$myLength = strlen($myCommand);
$myRequest = sprintf("%0".HEADERSZ."d%s\n", $myLength, $myCommand);
socket_write($socket, $myRequest);
$myOutput = "";
$myLength = socket_read($socket, HEADERSZ);
while (FALSE != ($line = @socket_read($socket, $myLength, PHP_BINARY_READ))) {
$myOutput = $myOutput . $line;
}
$bptr = strpos($myOutput, "(");
$eptr = strpos($myOutput, ")");
//assert($bptr !=0);
//assert($eptr !=0);
//assert($bptr < $eptr);
$myOutput = substr($myOutput, ($bptr+1), ($eptr-$bptr-1));
$myOutput = decryptString($myOutput);
return ($myOutput);
}
$command = "LIST VOC SAMPLE 3";
$output = serviceRequest($command);
//header("Content-type: text/plain");
echo "<pre>".$output."</pre>";
?>
3. Writing a backend Universe BASIC program
As I decided above, this backend program only needs to be able to base64 decode the command string and its arguments, and execute it. The output from the command is captured, cleaned up, and encoded to be returned to the frontend.
Note that all we have to do here is DISPLAY the response; u2pipe does the communication logic for us.
Let's name this program U2PIPERUN. Make sure you compile and catalog it in the appropriate account.
0001: DISPLAY "!":
0002:
0003:
0004: DEFFUN DECODE64(TEXT)
0005: DEFFUN ENCODE64(TEXT)
0006: EQU LF TO CHAR(10)
0007: EQU FF TO CHAR(12)
0008: EQU CR TO CHAR(13)
0009:
0010:
0011: COMMAND = FIELD(@SENTENCE, " ", 2, LEN(@SENTENCE))
0012: COMMAND = DECODE64(COMMAND)
0013:
0014:
0015: EXECUTE COMMAND CAPTURING OUTPUT RETURNING RETCODE
0016:
0017:
0018: OUTPUT = CONVERT(@FM, LF, OUTPUT)
0019: OUTPUT = CONVERT(CR, "", OUTPUT)
0020: OUTPUT = TRIM(OUTPUT, FF, "B")
0021: OUTPUT = TRIM(OUTPUT, LF, "B")
0022: OUTPUT = ENCODE64(OUTPUT)
0023:
0024:
0025: DISPLAY "(":OUTPUT:")"
0026: END
Bottom at line 26.
4. Setting the RUN parameter in u2pipe.ini
Your u2pipe.ini file should look something like this:
[U2PIPE] HOSTNAME=yourhost USER=webuser PASSWORD=webuser_password ACCOUNT=WEBACCOUNT RUN=U2PIPERUN
5. Testing your implementation
You should be ready to test your implementation. You may want to turn on the LOGFILE parameter in the u2pipe.ini file to see the data being passed back and forth. You may also want to add some kind of error logging to your U2PIPERUN program.
Other uses
Obviously, you don't want just anyone sending TCL commands to your Universe database.
ALSO, wininetd can be used for evil; googling on it reveals that it can be used as part of a Trojan-virus attack. Some anti-virus software may report it as such. Just like telnet and ftp, wininetd is only as safe as your firewall setup!
Why write a socket-based solution instead of just using UniObjects? In my case, I needed to implement database connectivity for a remotely hosted web site; I couldn't "wire" UniObjects directly into the web server. And I really didn't want to.
Here are some decisions I made when writing this utility.
In designing your own web or socket solution, I think it is important to keep all your business logic on the database server. In some examples of UniObjects programming, I see business logic migrated or duplicated on the client - bad, bad, bad.
Whether you use UniObjects or u2pipe (or SQL for that matter), design your user interface to be database independent. Instead of embedding data access and updates directly in your web page (or GUI screen), put your database access logic in a separate function library. Then call those functions from your web page. This way, you'll be able to easily write test scaffolding to test your web page, then swap in the "real" functions that actually hit your database.
You can download source code from http://xmailserver.org/wininetd.html
/***
* u2pipe.c
* universe request pipe utility
* CVS $Revision: 1.3 $ $Date: 2007/05/11 19:33:27 $
*
* Copyright (C) 2006 Rex Gozar
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
* http://www.gnu.org/licenses/lgpl.html
*
* Rex Gozar
* rkgozar@juno.com
***/
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <intcall.h>
#include "iniparser.h"
#define MAXBUF (256)
#define MAXBIGBUF (8192)
#ifndef TRUE
#define TRUE (1)
#endif
#ifndef FALSE
#define FALSE (0)
#endif
#define APP_NAME ("U2PIPE")
#define DEFAULT_VALUE ("")
#define REQUIRED (TRUE)
#define NOT_REQUIRED (FALSE)
/***
* NOTE - GETSLASHSZ must be smaller than REQUESTLENSZ
* and (GETSLASHSZ + REQUESTLENSZ + 1) must be smaller than MAXBUF!
***/
#define GETSLASH ("GET /")
#define GETSLASHSZ (5)
#define REQUESTLENSZ (6)
/***
* helper functions
***/
char * getTimeStamp(char *timestamp) {
time_t seconds = time(NULL);
struct tm *now = localtime(&seconds);
strftime(timestamp, MAXBUF, "%Y%m%d %H%M%S", now);
return timestamp;
}
/***
* main
*
* Note that the program should be compiled with __stdcall
* call convention so the intercall routines will properly link.
* Use __cdecl here to avoid a compiler warning.
***/
#ifndef WIN32
#define __cdecl
#endif
int __cdecl main(int argc, char *argv[]) {
time_t stime = time(NULL);
time_t etime = 0;
#ifdef WIN32
const char *WINDIR = getenv("WINDIR");
#else
const char *ETCDIR = "/etc";
#endif
const char *U2PIPE_INI = "u2pipe.ini";
long retcode = 0;
char buffer[MAXBUF] = "";
char timestamp[MAXBUF] = "";
const char *CLIENT_IP = getenv("CLIENT_IP");
const char *CLIENT_PORT = getenv("CLIENT_PORT");
char ipAddress[MAXBUF] = "";
char ipPort[MAXBUF] = "";
char iniFile[MAXBUF] = "";
dictionary *ini = NULL;
char *iniHostname = NULL;
char *iniUser = NULL;
char *iniPassword = NULL;
char *iniAccount = NULL;
char *iniRun = NULL;
char *iniLogfile = NULL;
char *iniAllowIP = NULL;
int logsw = FALSE;
FILE *fpLog;
int j = 0;
char *request = NULL;
long requestLen = 0;
char *response = NULL;
long responseLen = 0;
long responseMax = 0;
long sessionId = 0;
long code = 0;
long retcode2 = 0;
char *command = NULL;
long commandLen = 0;
/* check for NULL pointers and initialize ip address and port strings */
if (NULL != CLIENT_IP) {
strncpy(ipAddress, CLIENT_IP, strlen(CLIENT_IP));
}
if (NULL != CLIENT_PORT) {
strncpy(ipPort, CLIENT_PORT, strlen(CLIENT_PORT));
}
/* see if the ini file path is on the command line */
if (2 <= argc) {
if (MAXBUF < ( 1 + strlen(argv[1]) )) {
fprintf(stderr, "ini file pathname is longer than %i!\n", MAXBUF);
exit(1);
}
sprintf(iniFile, "%s", argv[1]);
} else {
#ifdef WIN32
if (NULL == WINDIR) {
fprintf(stderr, "WINDIR is NULL!\n");
exit(1);
}
/* build the default ini file path %WINDIR%\u2pipe.ini */
if (MAXBUF < ( 1 + strlen(WINDIR) + 1 + strlen(U2PIPE_INI) )) {
fprintf(stderr, "%%WINDIR%%\\%s is longer than %i!\n", U2PIPE_INI, MAXBUF);
exit(1);
}
sprintf(iniFile, "%s\\%s", WINDIR, U2PIPE_INI);
#else
if (MAXBUF < ( 1 + strlen(ETCDIR) + 1 + strlen(U2PIPE_INI) )) {
fprintf(stderr, "%s/%s is longer than %i!\n", ETCDIR, U2PIPE_INI, MAXBUF);
exit(1);
}
sprintf(iniFile, "%s/%s", ETCDIR, U2PIPE_INI);
#endif
}
/* parse config for user, password, account, etc */
ini = iniparser_new(iniFile);
iniHostname = iniparser_getstring(ini, "u2pipe:hostname", "");
iniUser = iniparser_getstring(ini, "u2pipe:user", "");
iniPassword = iniparser_getstring(ini, "u2pipe:password", "");
iniAccount = iniparser_getstring(ini, "u2pipe:account", "");
iniRun = iniparser_getstring(ini, "u2pipe:run", "");
iniLogfile = iniparser_getstring(ini, "u2pipe:logfile", "");
iniAllowIP = iniparser_getstring(ini, "u2pipe:allowIP", "");
#ifdef _DEBUG
fprintf(stderr, "iniFile=%s\n", iniFile);
fprintf(stderr, "ipAddress=%s\n", ipAddress);
fprintf(stderr, "ipPort=%s\n", ipPort);
fprintf(stderr, "iniFile=%s\n", iniFile);
fprintf(stderr, "iniHostname=%s\n", iniHostname);
fprintf(stderr, "iniUser=%s\n", iniUser);
fprintf(stderr, "iniPassword=%s\n", iniPassword);
fprintf(stderr, "iniAccount=%s\n", iniAccount);
fprintf(stderr, "iniRun=%s\n", iniRun);
fprintf(stderr, "iniLogfile=%s\n", iniLogfile);
fprintf(stderr, "iniAllowIP=%s\n", iniAllowIP);
#endif
/* open the log file */
logsw = (0 != strlen(iniLogfile));
if (logsw) {
fpLog = fopen(iniLogfile, "a+");
if (NULL == fpLog) {
fprintf(stderr, "unable to open log file %s\n", iniLogfile);
exit(1);
}
/* print a separator line for each request */
fprintf(fpLog, "****************************************\n");
}
/* log the requester */
if (logsw) {
fprintf(fpLog, "%s ipAddress=%s\n", getTimeStamp(timestamp), ipAddress);
fprintf(fpLog, "%s ipPort=%s\n", getTimeStamp(timestamp), ipPort);
}
/* check allowed ip-addresses */
if (0 != strlen(iniAllowIP)) {
/* sandwich the list between semi-colons */
if (MAXBUF < 3 + strlen(iniAllowIP)) {
if (logsw) {
fprintf(fpLog, "%s ALLOWIP list is too long!\n", getTimeStamp(timestamp));
}
exit(1);
}
sprintf(buffer, ";%s;", iniAllowIP);
memset(iniAllowIP, 0, MAXBUF);
strncpy(iniAllowIP, buffer, strlen(buffer));
/* do the same to the ip address */
if (MAXBUF < 3 + strlen(ipAddress)) {
if (logsw) {
fprintf(fpLog, "%s ipAddress is too long!\n", getTimeStamp(timestamp));
}
exit(1);
}
sprintf(buffer, ";%s;", ipAddress);
/* now look for the ipAddress within the allowed IP's */
if (NULL == strstr(iniAllowIP, buffer)) {
if (logsw) {
fprintf(fpLog, "%s ipAddress is not allowed!\n", getTimeStamp(timestamp));
}
exit(1);
}
}
/****
*
* requests are piped in via stdin, and must be formatted:
*
* GET /000009something
* or
* 000009something
*
* where:
* "GET /" is an optional literal string (typically passed by a web request.)
* "000009" is a six-digit request length.
* "something" is the actual request.
***/
/* look for "GET /" */
memset(buffer, 0, MAXBUF);
fread(buffer, sizeof(char), GETSLASHSZ, stdin);
if (0 == strncmp(buffer, GETSLASH, GETSLASHSZ)) {
/* discard the buffer contents and get REQUESTLENSZ more characters */
memset(buffer, 0, MAXBUF);
fread(buffer, sizeof(char), REQUESTLENSZ, stdin);
} else {
/* read in more characters to fill in the request length */
j = GETSLASHSZ;
while (j < REQUESTLENSZ) {
int c = fgetc(stdin);
if (EOF == c) {
if (logsw) {
fprintf(fpLog, "%s not enough characters in stdin!\n", getTimeStamp(timestamp));
}
exit(1);
}
buffer[j] = c;
j++;
}
}
/* check the request length string */
if (REQUESTLENSZ != strlen(buffer)) {
if (logsw) {
fprintf(fpLog, "%s request length is not %i characters!\n", getTimeStamp(timestamp), REQUESTLENSZ);
}
exit(1);
}
for (j = 0; j < REQUESTLENSZ; j++) {
if (!isdigit(buffer[j])) {
if (logsw) {
fprintf(fpLog, "%s request length is non-numeric!\n", getTimeStamp(timestamp));
}
exit(1);
}
}
requestLen = atoi(buffer);
if (0 == requestLen) {
if (logsw) {
fprintf(fpLog, "%s request length is zero!\n", getTimeStamp(timestamp));
}
exit(1);
}
/* get the memory to read in the rest of the request */
request = (char *) malloc(1 + requestLen);
if (NULL == request) {
if (logsw) {
fprintf(fpLog, "%s request malloc failed!\n", getTimeStamp(timestamp));
}
exit(1);
}
/* read in the request */
memset(request, 0, 1+requestLen);
fread(request, sizeof(char), requestLen, stdin);
/* log the request */
if (logsw) {
fprintf(fpLog, "%s REQUEST=%s\n", getTimeStamp(timestamp), request);
}
/*********************
* process the request
**********************/
sessionId = ic_opensession(iniHostname, iniUser, iniPassword, iniAccount, &code, NULL);
if (0 != code) {
if (logsw) {
fprintf(fpLog, "%s ic_opensession failed! %i\n", getTimeStamp(timestamp), code);
}
exit(1);
}
if (0 != strlen(iniRun)) {
/* add one to the command length for the space between */
commandLen = strlen(iniRun) + 1 + requestLen;
command = (char *) malloc(1+commandLen);
memset(command, 0, commandLen);
sprintf(command, "%s %s", iniRun, request);
} else {
commandLen = requestLen;
command = (char *) malloc(1+commandLen);
memset(command, 0, commandLen);
sprintf(command, "%s", request);
}
responseMax = MAXBIGBUF;
response = (char *) malloc(responseMax);
memset(response, 0, responseMax);
ic_execute(command, &commandLen, response, &responseMax,
&responseLen, &retcode, &retcode2, &code);
if (logsw) {
fprintf(fpLog, "%s IC_EXECUTE=%i\n", getTimeStamp(timestamp), code);
fprintf(fpLog, "%s responseLen=%i\n", getTimeStamp(timestamp), responseLen);
}
/* see if we need to expand the response buffer to hold all the data */
while (IE_BTS == code) {
char *tmpbuf = NULL;
long tmpmax = 0;
long tmplen = 0;
char *newbuf = NULL;
long newmax = 0;
/* create a temporary buffer to hold the next block of data */
tmpmax = responseMax;
tmpbuf = (char *) malloc(tmpmax);
if (NULL == tmpbuf) {
fprintf(stderr, "malloc failed when creating tmpbuf!\n");
}
memset(tmpbuf, 0, tmpmax);
tmplen = 0;
/* get the next block of data */
ic_executecontinue(tmpbuf, &tmpmax, &tmplen, &retcode, &retcode2, &code);
if (logsw) {
fprintf(fpLog, "%s IC_EXECUTECONTINUE=%i\n", getTimeStamp(timestamp), code);
}
/* create a new buffer to hold both the response and tmp buffers */
/* note that the new buffer is double the size of the previous iteration */
newmax = responseMax + tmpmax;
newbuf = (char *) malloc(newmax);
if (NULL == newbuf) {
fprintf(stderr, "malloc failed when creating newbuf!\n");
exit(1);
}
memset(newbuf, 0, newmax);
/* copy both the response and tmpbuf into the new buffer */
memcpy(newbuf, response, responseLen);
memcpy(newbuf + responseLen, tmpbuf, tmpmax);
/* release the memory for old data buffers and point response to the new data */
free(tmpbuf);
free(response);
response = newbuf;
responseMax = newmax;
responseLen = responseLen + tmplen;
}
ic_quit(&code);
/* log the response */
if (logsw) {
fprintf(fpLog, "%s RESPONSE=%s\n", getTimeStamp(timestamp), response);
fprintf(fpLog, "%s responseLen=%i\n", getTimeStamp(timestamp), responseLen);
fprintf(fpLog, "%s strlen(response)=%i\n", getTimeStamp(timestamp), strlen(response));
}
/* return the response on stdout */
fprintf(stdout, "%06i%s\n", responseLen, response);
fflush(stdout);
/* done */
if (logsw) {
etime = time(NULL) - stime;
fprintf(fpLog, "%s COMPLETED (%lu)\n", getTimeStamp(timestamp), etime);
fflush(fpLog);
fclose(fpLog);
}
exit(0);
}
/*
Based upon libiniparser, by Nicolas Devillard
Hacked into 1 file (m-iniparser) by Freek/2005
Original terms following:
-- -
Copyright (c) 2000 by Nicolas Devillard (ndevilla AT free DOT fr).
Written by Nicolas Devillard. Not derived from licensed software.
Permission is granted to anyone to use this software for any
purpose on any computer system, and to redistribute it freely,
subject to the following restrictions:
1. The author is not responsible for the consequences of use of
this software, no matter how awful, even if they arise
from defects in it.
2. The origin of this software must not be misrepresented, either
by explicit claim or by omission.
3. Altered versions must be plainly marked as such, and must not
be misrepresented as being the original software.
4. This notice may not be removed or altered.
*/
#ifndef _INIPARSER_H_
#define _INIPARSER_H_
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* #include <unistd.h> */
#include <ctype.h>
#ifdef __cplusplus
extern "C" {
#endif
typedef struct _dictionary_ {
/** Number of entries in dictionary */
int n;
/** Storage size */
int size;
/** List of string values */
char **val;
/** List of string keys */
char **key ;
/** List of hash values for keys */
unsigned *hash;
} dictionary ;
/* generated by genproto */
dictionary * iniparser_new(char *ininame);
void iniparser_free(dictionary *d);
int iniparser_getnsec(dictionary *d);
char * iniparser_getsecname(dictionary *d, int n);
void iniparser_dump(dictionary *d, FILE *f);
void iniparser_dump_ini(dictionary *d, FILE *f);
char * iniparser_getkey(dictionary *d, char *section, char *key);
char * iniparser_getstr(dictionary *d, char *key);
char * iniparser_getstring(dictionary *d, char *key, char *def);
int iniparser_getint(dictionary *d, char *key, int notfound);
double iniparser_getdouble(dictionary *d, char *key, double notfound);
int iniparser_getboolean(dictionary *d, char *key, int notfound);
int iniparser_find_entry(dictionary *ini, char *entry);
int iniparser_setstr(dictionary *ini, char *entry, char *val);
void iniparser_unset(dictionary *ini, char *entry);
#ifdef __cplusplus
}
#endif
#endif
/*
Based upon libiniparser, by Nicolas Devillard
Hacked into 1 file (m-iniparser) by Freek/2005
Original terms following:
-- -
Copyright (c) 2000 by Nicolas Devillard (ndevilla AT free DOT fr).
Written by Nicolas Devillard. Not derived from licensed software.
Permission is granted to anyone to use this software for any
purpose on any computer system, and to redistribute it freely,
subject to the following restrictions:
1. The author is not responsible for the consequences of use of
this software, no matter how awful, even if they arise
from defects in it.
2. The origin of this software must not be misrepresented, either
by explicit claim or by omission.
3. Altered versions must be plainly marked as such, and must not
be misrepresented as being the original software.
4. This notice may not be removed or altered.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* #include <unistd.h> */
#include "iniparser.h"
#ifdef __cplusplus
extern "C" {
#endif
/* strlib.c following */
#define ASCIILINESZ 1024
/*-------------------------------------------------------------------------*/
/**
@brief Convert a string to lowercase.
@param s String to convert.
@return ptr to statically allocated string.
This function returns a pointer to a statically allocated string
containing a lowercased version of the input string. Do not free
or modify the returned string! Since the returned string is statically
allocated, it will be modified at each function call (not re-entrant).
*/
/*--------------------------------------------------------------------------*/
static char * strlwc(char *s)
{
static char l[ASCIILINESZ+1];
int i ;
if (s==NULL) return NULL ;
memset(l, 0, ASCIILINESZ+1);
i=0 ;
while (s[i] && i<ASCIILINESZ) {
l[i] = (char)tolower((int)s[i]);
i++ ;
}
l[ASCIILINESZ]=(char)0;
return l ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Convert a string to uppercase.
@param s String to convert.
@return ptr to statically allocated string.
This function returns a pointer to a statically allocated string
containing an uppercased version of the input string. Do not free
or modify the returned string! Since the returned string is statically
allocated, it will be modified at each function call (not re-entrant).
*/
/*--------------------------------------------------------------------------*/
static char * strupc(char *s)
{
static char l[ASCIILINESZ+1];
int i ;
if (s==NULL) return NULL ;
memset(l, 0, ASCIILINESZ+1);
i=0 ;
while (s[i] && i<ASCIILINESZ) {
l[i] = (char)toupper((int)s[i]);
i++ ;
}
l[ASCIILINESZ]=(char)0;
return l ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Skip blanks until the first non-blank character.
@param s String to parse.
@return Pointer to char inside given string.
This function returns a pointer to the first non-blank character in the
given string.
*/
/*--------------------------------------------------------------------------*/
static char * strskp(char *s)
{
char * skip = s;
if (s==NULL) return NULL ;
while (isspace((int)*skip) && *skip) skip++;
return skip ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Remove blanks at the end of a string.
@param s String to parse.
@return ptr to statically allocated string.
This function returns a pointer to a statically allocated string,
which is identical to the input string, except that all blank
characters at the end of the string have been removed.
Do not free or modify the returned string! Since the returned string
is statically allocated, it will be modified at each function call
(not re-entrant).
*/
/*--------------------------------------------------------------------------*/
static char * strcrop(char *s)
{
static char l[ASCIILINESZ+1];
char * last ;
if (s==NULL) return NULL ;
memset(l, 0, ASCIILINESZ+1);
strcpy(l, s);
last = l + strlen(l);
while (last > l) {
if (!isspace((int)*(last-1)))
break ;
last -- ;
}
*last = (char)0;
return l ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Remove blanks at the beginning and the end of a string.
@param s String to parse.
@return ptr to statically allocated string.
This function returns a pointer to a statically allocated string,
which is identical to the input string, except that all blank
characters at the end and the beg. of the string have been removed.
Do not free or modify the returned string! Since the returned string
is statically allocated, it will be modified at each function call
(not re-entrant).
*/
/*--------------------------------------------------------------------------*/
static char * strstrip(char *s)
{
static char l[ASCIILINESZ+1];
char * last ;
if (s==NULL) return NULL ;
while (isspace((int)*s) && *s) s++;
memset(l, 0, ASCIILINESZ+1);
strcpy(l, s);
last = l + strlen(l);
while (last > l) {
if (!isspace((int)*(last-1)))
break ;
last -- ;
}
*last = (char)0;
return (char*)l ;
}
/* dictionary.c.c following */
/** Maximum value size for integers and doubles. */
#define MAXVALSZ 1024
/** Minimal allocated number of entries in a dictionary */
#define DICTMINSZ 128
/** Invalid key token */
#define DICT_INVALID_KEY ((char*)-1)
/*
Doubles the allocated size associated to a pointer
'size' is the current allocated size.
*/
static void * mem_double(void *ptr, int size)
{
void *newptr;
newptr = calloc(2*size, 1);
memcpy(newptr, ptr, size);
free(ptr);
return newptr ;
}
/*---------------------------------------------------------------------------
Function codes
---------------------------------------------------------------------------*/
/*-------------------------------------------------------------------------*/
/**
@brief Compute the hash key for a string.
@param key Character string to use for key.
@return 1 unsigned int on at least 32 bits.
This hash function has been taken from an Article in Dr Dobbs Journal.
This is normally a collision-free function, distributing keys evenly.
The key is stored anyway in the struct so that collision can be avoided
by comparing the key itself in last resort.
*/
/*--------------------------------------------------------------------------*/
static unsigned dictionary_hash(char *key)
{
int len ;
unsigned hash ;
int i ;
len = strlen(key);
for (hash=0, i=0 ; i<len ; i++) {
hash += (unsigned)key[i] ;
hash += (hash<<10);
hash ^= (hash>>6) ;
}
hash += (hash <<3);
hash ^= (hash >>11);
hash += (hash <<15);
return hash ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Create a new dictionary object.
@param size Optional initial size of the dictionary.
@return 1 newly allocated dictionary objet.
This function allocates a new dictionary object of given size and returns
it. If you do not know in advance (roughly) the number of entries in the
dictionary, give size=0.
*/
/*--------------------------------------------------------------------------*/
static dictionary * dictionary_new(int size)
{
dictionary *d ;
/* If no size was specified, allocate space for DICTMINSZ */
if (size<DICTMINSZ) size=DICTMINSZ ;
d = (dictionary *)calloc(1, sizeof(dictionary));
d->size = size ;
d->val = (char **)calloc(size, sizeof(char*));
d->key = (char **)calloc(size, sizeof(char*));
d->hash = (unsigned int *)calloc(size, sizeof(unsigned));
return d;
}
/*-------------------------------------------------------------------------*/
/**
@brief Delete a dictionary object
@param d dictionary object to deallocate.
@return void
Deallocate a dictionary object and all memory associated to it.
*/
/*--------------------------------------------------------------------------*/
static void dictionary_del(dictionary *d)
{
int i ;
if (d==NULL) return ;
for (i=0 ; i<d->size ; i++) {
if (d->key[i]!=NULL)
free(d->key[i]);
if (d->val[i]!=NULL)
free(d->val[i]);
}
free(d->val);
free(d->key);
free(d->hash);
free(d);
return;
}
/*-------------------------------------------------------------------------*/
/**
@brief Get a value from a dictionary.
@param d dictionary object to search.
@param key Key to look for in the dictionary.
@param def Default value to return if key not found.
@return 1 pointer to internally allocated character string.
This function locates a key in a dictionary and returns a pointer to its
value, or the passed 'def' pointer if no such key can be found in
dictionary. The returned character pointer points to data internal to the
dictionary object, you should not try to free it or modify it.
*/
/*--------------------------------------------------------------------------*/
static char * dictionary_get(dictionary *d, char *key, char *def)
{
unsigned hash ;
int i ;
hash = dictionary_hash(key);
for (i=0 ; i<d->size ; i++) {
if (d->key==NULL)
continue ;
/* Compare hash */
if (hash==d->hash[i]) {
/* Compare string, to avoid hash collisions */
if (!strcmp(key, d->key[i])) {
return d->val[i] ;
}
}
}
return def ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Set a value in a dictionary.
@param d dictionary object to modify.
@param key Key to modify or add.
@param val Value to add.
@return void
If the given key is found in the dictionary, the associated value is
replaced by the provided one. If the key cannot be found in the
dictionary, it is added to it.
It is Ok to provide a NULL value for val, but NULL values for the dictionary
or the key are considered as errors: the function will return immediately
in such a case.
Notice that if you dictionary_set a variable to NULL, a call to
dictionary_get will return a NULL value: the variable will be found, and
its value (NULL) is returned. In other words, setting the variable
content to NULL is equivalent to deleting the variable from the
dictionary. It is not possible (in this implementation) to have a key in
the dictionary without value.
*/
/*--------------------------------------------------------------------------*/
static void dictionary_set(dictionary *d, char *key, char *val)
{
int i ;
unsigned hash ;
if (d==NULL || key==NULL) return ;
/* Compute hash for this key */
hash = dictionary_hash(key) ;
/* Find if value is already in blackboard */
if (d->n>0) {
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL)
continue ;
if (hash==d->hash[i]) { /* Same hash value */
if (!strcmp(key, d->key[i])) { /* Same key */
/* Found a value: modify and return */
if (d->val[i]!=NULL)
free(d->val[i]);
d->val[i] = val ? strdup(val) : NULL ;
/* Value has been modified: return */
return ;
}
}
}
}
/* Add a new value */
/* See if dictionary needs to grow */
if (d->n==d->size) {
/* Reached maximum size: reallocate blackboard */
d->val = (char **)mem_double(d->val, d->size * sizeof(char*)) ;
d->key = (char **)mem_double(d->key, d->size * sizeof(char*)) ;
d->hash = (unsigned int *)mem_double(d->hash, d->size * sizeof(unsigned)) ;
/* Double size */
d->size *= 2 ;
}
/* Insert key in the first empty slot */
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL) {
/* Add key here */
break ;
}
}
/* Copy key */
d->key[i] = strdup(key);
d->val[i] = val ? strdup(val) : NULL ;
d->hash[i] = hash;
d->n ++ ;
return ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Delete a key in a dictionary
@param d dictionary object to modify.
@param key Key to remove.
@return void
This function deletes a key in a dictionary. Nothing is done if the
key cannot be found.
*/
/*--------------------------------------------------------------------------*/
static void dictionary_unset(dictionary *d, char *key)
{
unsigned hash ;
int i ;
hash = dictionary_hash(key);
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL)
continue ;
/* Compare hash */
if (hash==d->hash[i]) {
/* Compare string, to avoid hash collisions */
if (!strcmp(key, d->key[i])) {
/* Found key */
break ;
}
}
}
if (i>=d->size)
/* Key not found */
return ;
free(d->key[i]);
d->key[i] = NULL ;
if (d->val[i]!=NULL) {
free(d->val[i]);
d->val[i] = NULL ;
}
d->hash[i] = 0 ;
d->n -- ;
return ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Dump a dictionary to an opened file pointer.
@param d Dictionary to dump
@param f Opened file pointer.
@return void
Dumps a dictionary onto an opened file pointer. Key pairs are printed out
as @c [Key]=[Value], one per line. It is Ok to provide stdout or stderr as
output file pointers.
*/
/*--------------------------------------------------------------------------*/
static void dictionary_dump(dictionary *d, FILE *f)
{
int i;
if (d==NULL || f==NULL) return;
for (i=0; i<d->size; i++) {
if (d->key[i] == NULL)
continue ;
if (d->val[i] != NULL) {
fprintf(f, "[%s]=[%s]\n", d->key[i], d->val[i]);
} else {
fprintf(f, "[%s]=UNDEF\n", d->key[i]);
}
}
return;
}
/* iniparser.c.c following */
#define ASCIILINESZ 1024
#define INI_INVALID_KEY ((char*)-1)
/* Private: add an entry to the dictionary */
static void iniparser_add_entry(
dictionary *d,
char *sec,
char *key,
char *val)
{
char longkey[2*ASCIILINESZ+1];
/* Make a key as section:keyword */
if (key!=NULL) {
sprintf(longkey, "%s:%s", sec, key);
} else {
strcpy(longkey, sec);
}
/* Add (key,val) to dictionary */
dictionary_set(d, longkey, val);
return ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Get number of sections in a dictionary
@param d Dictionary to examine
@return int Number of sections found in dictionary
This function returns the number of sections found in a dictionary.
The test to recognize sections is done on the string stored in the
dictionary: a section name is given as "section" whereas a key is
stored as "section:key", thus the test looks for entries that do not
contain a colon.
This clearly fails in the case a section name contains a colon, but
this should simply be avoided.
This function returns -1 in case of error.
*/
/*--------------------------------------------------------------------------*/
int iniparser_getnsec(dictionary *d)
{
int i ;
int nsec ;
if (d==NULL) return -1 ;
nsec=0 ;
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL)
continue ;
if (strchr(d->key[i], ':')==NULL) {
nsec ++ ;
}
}
return nsec ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Get name for section n in a dictionary.
@param d Dictionary to examine
@param n Section number (from 0 to nsec-1).
@return Pointer to char string
This function locates the n-th section in a dictionary and returns
its name as a pointer to a string statically allocated inside the
dictionary. Do not free or modify the returned string!
This function returns NULL in case of error.
*/
/*--------------------------------------------------------------------------*/
char * iniparser_getsecname(dictionary *d, int n)
{
int i ;
int foundsec ;
if (d==NULL || n<0) return NULL ;
foundsec=0 ;
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL)
continue ;
if (strchr(d->key[i], ':')==NULL) {
foundsec++ ;
if (foundsec>n)
break ;
}
}
if (foundsec<=n) {
return NULL ;
}
return d->key[i] ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Dump a dictionary to an opened file pointer.
@param d Dictionary to dump.
@param f Opened file pointer to dump to.
@return void
This function prints out the contents of a dictionary, one element by
line, onto the provided file pointer. It is OK to specify @c stderr
or @c stdout as output files. This function is meant for debugging
purposes mostly.
*/
/*--------------------------------------------------------------------------*/
void iniparser_dump(dictionary *d, FILE *f)
{
dictionary_dump(d,f);
}
/*-------------------------------------------------------------------------*/
/**
@brief Save a dictionary to a loadable ini file
@param d Dictionary to dump
@param f Opened file pointer to dump to
@return void
This function dumps a given dictionary into a loadable ini file.
It is Ok to specify @c stderr or @c stdout as output files.
*/
/*--------------------------------------------------------------------------*/
void iniparser_dump_ini(dictionary *d, FILE *f)
{
int i, j ;
char keym[ASCIILINESZ+1];
int nsec ;
char * secname ;
int seclen ;
if (d==NULL || f==NULL) return ;
nsec = iniparser_getnsec(d);
if (nsec<1) {
/* No section in file: dump all keys as they are */
for (i=0 ; i<d->size ; i++) {
if (d->key[i]==NULL)
continue ;
fprintf(f, "%s = %s\n", d->key[i], d->val[i]);
}
return ;
}
for (i=0 ; i<nsec ; i++) {
secname = iniparser_getsecname(d, i) ;
seclen = (int)strlen(secname);
fprintf(f, "\n[%s]\n", secname);
sprintf(keym, "%s:", secname);
for (j=0 ; j<d->size ; j++) {
if (d->key[j]==NULL)
continue ;
if (!strncmp(d->key[j], keym, seclen+1)) {
fprintf(f,
"%-30s = %s\n",
d->key[j]+seclen+1,
d->val[j] ? d->val[j] : "");
}
}
}
fprintf(f, "\n");
return ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Get the string associated to a key, return NULL if not found
@param d Dictionary to search
@param key Key string to look for
@return pointer to statically allocated character string, or NULL.
This function queries a dictionary for a key. A key as read from an
ini file is given as "section:key". If the key cannot be found,
NULL is returned.
The returned char pointer is pointing to a string allocated in
the dictionary, do not free or modify it.
This function is only provided for backwards compatibility with
previous versions of iniparser. It is recommended to use
iniparser_getstring() instead.
*/
/*--------------------------------------------------------------------------*/
char * iniparser_getstr(dictionary *d, char *key)
{
return iniparser_getstring(d, key, NULL);
}
/*-------------------------------------------------------------------------*/
/**
@brief Get the string associated to a key
@param d Dictionary to search
@param key Key string to look for
@param def Default value to return if key not found.
@return pointer to statically allocated character string
This function queries a dictionary for a key. A key as read from an
ini file is given as "section:key". If the key cannot be found,
the pointer passed as 'def' is returned.
The returned char pointer is pointing to a string allocated in
the dictionary, do not free or modify it.
*/
/*--------------------------------------------------------------------------*/
char * iniparser_getstring(dictionary *d, char *key, char *def)
{
char * lc_key ;
char * sval ;
if (d==NULL || key==NULL)
return def ;
lc_key = strdup(strlwc(key));
sval = dictionary_get(d, lc_key, def);
free(lc_key);
return sval ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Get the string associated to a key, convert to an int
@param d Dictionary to search
@param key Key string to look for
@param notfound Value to return in case of error
@return integer
This function queries a dictionary for a key. A key as read from an
ini file is given as "section:key". If the key cannot be found,
the notfound value is returned.
*/
/*--------------------------------------------------------------------------*/
int iniparser_getint(dictionary *d, char *key, int notfound)
{
char *str ;
str = iniparser_getstring(d, key, INI_INVALID_KEY);
if (str==INI_INVALID_KEY) return notfound ;
return atoi(str);
}
/*-------------------------------------------------------------------------*/
/**
@brief Get the string associated to a key, convert to a double
@param d Dictionary to search
@param key Key string to look for
@param notfound Value to return in case of error
@return double
This function queries a dictionary for a key. A key as read from an
ini file is given as "section:key". If the key cannot be found,
the notfound value is returned.
*/
/*--------------------------------------------------------------------------*/
double iniparser_getdouble(dictionary *d, char *key, double notfound)
{
char * str ;
str = iniparser_getstring(d, key, INI_INVALID_KEY);
if (str==INI_INVALID_KEY) return notfound ;
return atof(str);
}
/*-------------------------------------------------------------------------*/
/**
@brief Get the string associated to a key, convert to a boolean
@param d Dictionary to search
@param key Key string to look for
@param notfound Value to return in case of error
@return integer
This function queries a dictionary for a key. A key as read from an
ini file is given as "section:key". If the key cannot be found,
the notfound value is returned.
A true boolean is found if one of the following is matched:
- A string starting with 'y'
- A string starting with 'Y'
- A string starting with 't'
- A string starting with 'T'
- A string starting with '1'
A false boolean is found if one of the following is matched:
- A string starting with 'n'
- A string starting with 'N'
- A string starting with 'f'
- A string starting with 'F'
- A string starting with '0'
The notfound value returned if no boolean is identified, does not
necessarily have to be 0 or 1.
*/
/*--------------------------------------------------------------------------*/
int iniparser_getboolean(dictionary *d, char *key, int notfound)
{
char *c ;
int ret ;
c = iniparser_getstring(d, key, INI_INVALID_KEY);
if (c==INI_INVALID_KEY) return notfound ;
if (c[0]=='y' || c[0]=='Y' || c[0]=='1' || c[0]=='t' || c[0]=='T') {
ret = 1 ;
} else if (c[0]=='n' || c[0]=='N' || c[0]=='0' || c[0]=='f' || c[0]=='F') {
ret = 0 ;
} else {
ret = notfound ;
}
return ret;
}
/*-------------------------------------------------------------------------*/
/**
@brief Finds out if a given entry exists in a dictionary
@param ini Dictionary to search
@param entry Name of the entry to look for
@return integer 1 if entry exists, 0 otherwise
Finds out if a given entry exists in the dictionary. Since sections
are stored as keys with NULL associated values, this is the only way
of querying for the presence of sections in a dictionary.
*/
/*--------------------------------------------------------------------------*/
int iniparser_find_entry(
dictionary *ini,
char *entry
)
{
int found=0 ;
if (iniparser_getstring(ini, entry, INI_INVALID_KEY)!=INI_INVALID_KEY) {
found = 1 ;
}
return found ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Set an entry in a dictionary.
@param ini Dictionary to modify.
@param entry Entry to modify (entry name)
@param val New value to associate to the entry.
@return int 0 if Ok, -1 otherwise.
If the given entry can be found in the dictionary, it is modified to
contain the provided value. If it cannot be found, -1 is returned.
It is Ok to set val to NULL.
*/
/*--------------------------------------------------------------------------*/
int iniparser_setstr(dictionary *ini, char *entry, char *val)
{
dictionary_set(ini, strlwc(entry), val);
return 0 ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Delete an entry in a dictionary
@param ini Dictionary to modify
@param entry Entry to delete (entry name)
@return void
If the given entry can be found, it is deleted from the dictionary.
*/
/*--------------------------------------------------------------------------*/
void iniparser_unset(dictionary *ini, char *entry)
{
dictionary_unset(ini, strlwc(entry));
}
/*-------------------------------------------------------------------------*/
/**
@brief Parse an ini file and return an allocated dictionary object
@param ininame Name of the ini file to read.
@return Pointer to newly allocated dictionary
This is the parser for ini files. This function is called, providing
the name of the file to be read. It returns a dictionary object that
should not be accessed directly, but through accessor functions
instead.
The returned dictionary must be freed using iniparser_free().
*/
/*--------------------------------------------------------------------------*/
dictionary * iniparser_new(char *ininame)
{
dictionary *d ;
char lin[ASCIILINESZ+1];
char sec[ASCIILINESZ+1];
char key[ASCIILINESZ+1];
char val[ASCIILINESZ+1];
char *where ;
FILE *ini ;
int lineno ;
if ((ini=fopen(ininame, "r"))==NULL) {
return NULL ;
}
sec[0]=0;
/*
* Initialize a new dictionary entry
*/
d = dictionary_new(0);
lineno = 0 ;
while (fgets(lin, ASCIILINESZ, ini)!=NULL) {
lineno++ ;
where = strskp(lin); /* Skip leading spaces */
if (*where==';' || *where=='#' || *where==0)
continue ; /* Comment lines */
else {
if (sscanf(where, "[%[^]]", sec)==1) {
/* Valid section name */
strcpy(sec, strlwc(sec));
iniparser_add_entry(d, sec, NULL, NULL);
} else if (sscanf (where, "%[^=] = \"%[^\"]\"", key, val) == 2
|| sscanf (where, "%[^=] = '%[^\']'", key, val) == 2
|| sscanf (where, "%[^=] = %[^;#]", key, val) == 2) {
strcpy(key, strlwc(strcrop(key)));
/*
* sscanf cannot handle "" or '' as empty value,
* this is done here
*/
if (!strcmp(val, "\"\"") || !strcmp(val, "''")) {
val[0] = (char)0;
} else {
strcpy(val, strcrop(val));
}
iniparser_add_entry(d, sec, key, val);
}
}
}
fclose(ini);
return d ;
}
/*-------------------------------------------------------------------------*/
/**
@brief Free all memory associated to an ini dictionary
@param d Dictionary to free
@return void
Free all memory associated to an ini dictionary.
It is mandatory to call this function before the dictionary object
gets out of the current context.
*/
/*--------------------------------------------------------------------------*/
void iniparser_free(dictionary *d)
{
dictionary_del(d);
}
#ifdef __cplusplus
}
#endif
http://www.gnu.org/licenses/lgpl.html