mirror of
https://github.com/samba-team/samba.git
synced 2025-12-14 20:23:54 +03:00
382 lines
9.3 KiB
C
382 lines
9.3 KiB
C
/*
|
|
* @file ejsString.c
|
|
* @brief EJScript string class
|
|
*/
|
|
/********************************* Copyright **********************************/
|
|
/*
|
|
* @copy default
|
|
*
|
|
* Copyright (c) Mbedthis Software LLC, 2003-2006. All Rights Reserved.
|
|
* Copyright (c) Michael O'Brien, 1994-1995. All Rights Reserved.
|
|
*
|
|
* This software is distributed under commercial and open source licenses.
|
|
* You may use the GPL open source license described below or you may acquire
|
|
* a commercial license from Mbedthis Software. You agree to be fully bound
|
|
* by the terms of either license. Consult the LICENSE.TXT distributed with
|
|
* this software for full details.
|
|
*
|
|
* This software is open source; 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. See the GNU General Public License for more
|
|
* details at: http://www.mbedthis.com/downloads/gplLicense.html
|
|
*
|
|
* This program is distributed WITHOUT ANY WARRANTY; without even the
|
|
* implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
*
|
|
* This GPL license does NOT permit incorporating this software into
|
|
* proprietary programs. If you are unable to comply with the GPL, you must
|
|
* acquire a commercial license to use this software. Commercial licenses
|
|
* for this software and support services are available from Mbedthis
|
|
* Software at http://www.mbedthis.com
|
|
*
|
|
* @end
|
|
*/
|
|
/********************************** Includes **********************************/
|
|
|
|
#include "ejs.h"
|
|
|
|
#if BLD_FEATURE_EJS
|
|
/******************************************************************************/
|
|
/*********************************** Constructors *****************************/
|
|
/******************************************************************************/
|
|
/*
|
|
* String constructor.
|
|
*/
|
|
|
|
int ejsStringConstructor(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
char *str;
|
|
|
|
if (argc == 0) {
|
|
ejsSetReturnValueToString(ejs, "");
|
|
|
|
} else if (argc == 1) {
|
|
/* MOB -- rc */
|
|
str = ejsVarToString(ejs, argv[0]);
|
|
ejsSetReturnValueToString(ejs, str);
|
|
|
|
} else {
|
|
ejsArgError(ejs, "usage: String([var])");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/******************************** Visible Methods *****************************/
|
|
/******************************************************************************/
|
|
/*
|
|
* Return a string containing the character at a given index
|
|
*
|
|
* String string.charAt(Number)
|
|
*/
|
|
|
|
static int charAt(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
EjsNum num;
|
|
char buf[2];
|
|
|
|
if (argc != 1) {
|
|
ejsArgError(ejs, "usage: charAt(integer)");
|
|
return -1;
|
|
}
|
|
|
|
num = ejsVarToNumber(argv[0]);
|
|
if (num < 0 || num >= thisObj->length) {
|
|
ejsError(ejs, EJS_RANGE_ERROR, "Bad index");
|
|
return -1;
|
|
}
|
|
|
|
mprAssert(ejsVarIsString(thisObj));
|
|
|
|
buf[0] = argv[0]->string[num];
|
|
buf[1] = '\0';
|
|
ejsSetReturnValueToString(ejs, buf);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Return an integer containing the character at a given index
|
|
*
|
|
* Number string.charCodeAt(Number)
|
|
*/
|
|
|
|
static EjsNum charCodeAt(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
EjsNum num;
|
|
|
|
if (argc != 1) {
|
|
ejsArgError(ejs, "usage: charCodeAt(integer)");
|
|
return -1;
|
|
}
|
|
|
|
num = ejsVarToNumber(argv[0]);
|
|
if (num < 0 || num >= thisObj->length) {
|
|
ejsError(ejs, EJS_RANGE_ERROR, "Bad index");
|
|
return -1;
|
|
}
|
|
ejsSetReturnValueToNumber(ejs, (EjsNum) argv[0]->string[num]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Catenate
|
|
*
|
|
* String string.catenate(var, ...)
|
|
*/
|
|
|
|
static int concat(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
int i;
|
|
|
|
if (argc == 0) {
|
|
ejsArgError(ejs, "usage: concat(String, ...)");
|
|
return -1;
|
|
}
|
|
|
|
mprAssert(ejsVarIsString(thisObj));
|
|
|
|
for (i = 0; i < argc; i++) {
|
|
if (ejsStrcat(ejs, thisObj, argv[i]) < 0) {
|
|
ejsMemoryError(ejs);
|
|
return -1;
|
|
}
|
|
}
|
|
ejsSetReturnValue(ejs, thisObj);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Return the position of the first occurance of a substring
|
|
*
|
|
* Number string.indexOf(String subString [, Number start])
|
|
*/
|
|
|
|
static int indexOf(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
char *pat, *s1, *s2, *origin;
|
|
int start, i;
|
|
|
|
if (argc == 0 || argc > 2) {
|
|
ejsArgError(ejs, "usage: indexOf(String [, Number])");
|
|
return -1;
|
|
}
|
|
|
|
pat = ejsVarToString(ejs, argv[0]);
|
|
|
|
if (argc == 2) {
|
|
start = ejsVarToNumber(argv[1]);
|
|
if (start > thisObj->length) {
|
|
start = thisObj->length;
|
|
}
|
|
} else {
|
|
start = 0;
|
|
}
|
|
|
|
i = start;
|
|
for (origin = &thisObj->string[i]; i < thisObj->length; i++, origin++) {
|
|
s1 = origin;
|
|
for (s2 = pat; *s1 && *s2; s1++, s2++) {
|
|
if (*s1 != *s2) {
|
|
break;
|
|
}
|
|
}
|
|
if (*s2 == '\0') {
|
|
ejsSetReturnValueToNumber(ejs, (EjsNum) (origin - thisObj->string));
|
|
}
|
|
}
|
|
|
|
ejsSetReturnValueToNumber(ejs, (EjsNum) -1);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Return the position of the last occurance of a substring
|
|
*
|
|
* Number string.lastIndexOf(String subString [, Number start])
|
|
*/
|
|
|
|
static int lastIndexOf(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
char *pat, *s1, *s2, *origin;
|
|
int start;
|
|
|
|
if (argc == 0 || argc > 2) {
|
|
ejsArgError(ejs, "usage: indexOf(String [, Number])");
|
|
return -1;
|
|
}
|
|
|
|
pat = ejsVarToString(ejs, argv[0]);
|
|
|
|
if (argc == 2) {
|
|
start = ejsVarToNumber(argv[1]);
|
|
if (start > thisObj->length) {
|
|
start = thisObj->length;
|
|
}
|
|
} else {
|
|
start = 0;
|
|
}
|
|
|
|
origin = &thisObj->string[thisObj->length - 1];
|
|
for (; origin >= &thisObj->string[start]; origin--) {
|
|
|
|
s1 = origin;
|
|
for (s2 = pat; *s1 && *s2; s1++, s2++) {
|
|
if (*s1 != *s2) {
|
|
break;
|
|
}
|
|
}
|
|
if (*s2 == '\0') {
|
|
ejsSetReturnValueToNumber(ejs, (EjsNum) (origin - thisObj->string));
|
|
}
|
|
}
|
|
|
|
ejsSetReturnValueToNumber(ejs, (EjsNum) -1);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Return a substring
|
|
*
|
|
* Number string.slice(Number start, Number end)
|
|
*/
|
|
|
|
static int slice(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
EjsNum start, end;
|
|
|
|
if (argc != 2) {
|
|
ejsArgError(ejs, "usage: slice(Number, Number)");
|
|
return -1;
|
|
}
|
|
|
|
start = ejsVarToNumber(argv[0]);
|
|
end = ejsVarToNumber(argv[1]);
|
|
if (start < 0 || start >= thisObj->length) {
|
|
ejsError(ejs, EJS_RANGE_ERROR, "Bad start index");
|
|
return-1;
|
|
}
|
|
if (end < 0 || end >= thisObj->length) {
|
|
ejsError(ejs, EJS_RANGE_ERROR, "Bad end index");
|
|
return -1;
|
|
}
|
|
|
|
mprAssert(ejsVarIsString(thisObj));
|
|
|
|
ejsSetReturnValueToBinaryString(ejs, (uchar*) &thisObj->string[start],
|
|
end - start);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Split a string
|
|
*
|
|
* Number string.split(String delimiter [, Number limit])
|
|
*/
|
|
|
|
static int split(Ejs *ejs, EjsVar *thisObj, int argc, EjsVar **argv)
|
|
{
|
|
EjsVar *array, *vp;
|
|
char *delim, *last, *cp;
|
|
int len, limit, alloc;
|
|
|
|
if (argc == 0 || argc > 2) {
|
|
ejsArgError(ejs, "usage: split(String [, Number])");
|
|
return -1;
|
|
}
|
|
|
|
delim = ejsVarToStringEx(ejs, argv[0], &alloc);
|
|
|
|
limit = ejsVarToNumber(argv[1]);
|
|
|
|
array = ejsCreateArray(ejs, 0);
|
|
|
|
len = strlen(delim);
|
|
|
|
last = thisObj->string;
|
|
for (cp = last; *cp; cp++) {
|
|
if (*cp == *delim && strncmp(cp, delim, len) == 0) {
|
|
if (cp > last) {
|
|
vp = ejsCreateBinaryStringVar(ejs, (uchar*) last, (cp - last));
|
|
ejsAddArrayElt(ejs, array, vp, EJS_SHALLOW_COPY);
|
|
ejsFreeVar(ejs, vp);
|
|
}
|
|
}
|
|
}
|
|
|
|
ejsSetReturnValue(ejs, array);
|
|
ejsFreeVar(ejs, array);
|
|
|
|
if (alloc) {
|
|
mprFree(delim);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
* Create the object class
|
|
*/
|
|
|
|
int ejsDefineStringClass(Ejs *ejs)
|
|
{
|
|
EjsVar *sc;
|
|
|
|
sc = ejsDefineClass(ejs, "String", "Object", ejsStringConstructor);
|
|
if (sc == 0) {
|
|
return MPR_ERR_CANT_INITIALIZE;
|
|
}
|
|
|
|
ejsDefineCMethod(ejs, sc, "charAt", charAt, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "charCodeAt", charCodeAt, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "concat", concat, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "indexOf", indexOf, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "lastIndexOf", lastIndexOf, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "slice", slice, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "split", split, EJS_NO_LOCAL);
|
|
#if UNUSED
|
|
ejsDefineCMethod(ejs, sc, "match", match, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "replace", replace, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "search", search, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "substring", substring, EJS_NO_LOCAL);
|
|
// MOB bad name
|
|
ejsDefineCMethod(ejs, sc, "substr", substr, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "toLowerCase", toLowerCase, EJS_NO_LOCAL);
|
|
ejsDefineCMethod(ejs, sc, "toUpperCase", toUpperCase, EJS_NO_LOCAL);
|
|
|
|
// Static method
|
|
ejsDefineCMethod(ejs, sc, "fromCharCode", fromCharCode, 0, EJS_NO_LOCAL);
|
|
#endif
|
|
|
|
if (ejsObjHasErrors(sc)) {
|
|
ejsFreeVar(ejs, sc);
|
|
return MPR_ERR_CANT_CREATE;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
#endif /* BLD_FEATURE_EJS */
|
|
|
|
/*
|
|
* Local variables:
|
|
* tab-width: 4
|
|
* c-basic-offset: 4
|
|
* End:
|
|
* vim:tw=78
|
|
* vim600: sw=4 ts=4 fdm=marker
|
|
* vim<600: sw=4 ts=4
|
|
*/
|