1

私は何か間違ったことをしているに違いありません...または、これはYAJLのバグかもしれませんが、それは非常に疑わしいです。json オブジェクトから最初の要素を取得できません。YAJL ソースに戻って、parse_config.c の例でこれをテストしましたが、これも失敗しました。

sample.config の使用

/*
 * The configuration file for Yahoo! BrowserPlus, included in the YAJL
 * tree as a sample configuration file for parsing.
 *
 * This is the configuration file for BrowserPlus
 */

{
    // The type of build this is, which is accessible to JavaScript via
    // BrowserPlus.getPlatformInfo(); 
    // Different build types should only differ in signatures accepted
    // (BrowserPlus.crt) and configured distribution servers.
    "BuildType": "ephemeral",

    // the base url for the "primary" distribution server.  This server will
    // be the single source of truth for Permissions, and will used to 
    // attain services
    "DistServer": "http://browserplus.yahoo.com",

    // An array of "secondary" distribution servers, which will be checked
    // in order for services if the primary server has no components
    // available which match an issued require statement.
    "SecondaryDistServers": [
      "http://first.fictional.server",
      "http://second.fictional.server"
    ],

    // Logging Setup
    "Logging" :
    {
        // Log level.  Values: "debug"|"info"|"warn"|"error"|"fatal"|"off"
        "level": "BP_LOG_LEVEL",

        // Destination.  Values: "file"|"console"|"win32"
        "dest": "BP_LOG_DEST",

        // Log message layout.  Values: "standard"|"source"|"raw"
        "layout": "standard",

        // Time format.  Values: "utc"|"local"|"msec"
        "timeFormat": "utc",

        // File size in KB which will trigger a rollover
        "fileRolloverKB": 2048,

        // Whether to send file logging from each service to a distinct file.
        // Values: "combined"|"separate"
        "serviceLogMode": "combined"
    },

    // Daemon setup
    // Syntax: "Options": "option1 option2 etc"
    // -fg        run in foreground, log to console
    "Options":"",

    // Auto-shutdown daemon if idle for this time.  Use 0 for no auto-shutdown.
    "MaxIdleSecs": 5,

    // At the end of each BrowserPlus session a small web request is made
    // to yahoo to indicate that BrowserPlus was used.  This report includes
    // * information about the browser being used
    // * an "installation id", which is a unique token that's generated
    //   the first time BrowserPlus runs.
    //
    // By design, there is *no information* in this request that gives
    // Yahoo! information about:
    //   a) the site that the user is visiting (see, "url": false)
    //   b) who the user is (the installation token cannot be tracked to a
    //      specific user).
    //
    // This information is primarily captured to help Yahoo! understand
    // adoption and usage of the BrowserPlus platform.
    "UsageReporting":
    {
       "enabled": true,
       "url": false,
       "id": true
    },

    // "Breakpoints" is an array of strings holding named breakpoints.
    // Platform code checks for specific entries at certain key points, and if 
    // a matching entry is found here a DebugBreak will be performed.
    // For developers with Visual Studio installed, the DebugBreak will cause an
    // opportunity to perform just-in-time attachment of an existing or new 
    // debugger instance.
    // The currently-defined breakpoints are listed below:
    //      runServiceProcess - A DebugBreak is performed in the service 
    //                          "harness" just prior to service load.
    //      ax.FinalConstruct - A DebugBreak is performed at entry to 
    //                          FinalConstruct of the ActiveX plugin.
    //      PluginInit -        Very early in the NPAPI plugin initialization.
    //                          A wonderful spot to stop and set more
    //                          breakpoints.
    //"Breakpoints": ["runServiceProcess"],

    // How often we check for service updates.  We guarantee at least this
    // much time will pass between checks, though the true time may be
    // much more if sites which use browserplus are not visited.
    // The time is in seconds.
    "ServiceUpdatePollPeriod": 86400
}

"BuildType" <-- JSON オブジェクトの最初の要素を取得しようとしました。

これを行うためにparse_config.cファイルを変更しました...コードは次のとおりです。

int
main(void)
{
    size_t rd;
    yajl_val node;
    char errbuf[1024];

    /* null plug buffers */
    fileData[0] = errbuf[0] = 0;

    /* read the entire config file */
    rd = fread((void *) fileData, 1, sizeof(fileData) - 1, stdin);

    /* file read error handling */
    if (rd == 0 && !feof(stdin)) {
        fprintf(stderr, "error encountered on file read\n");
        return 1;
    } else if (rd >= sizeof(fileData) - 1) {
        fprintf(stderr, "config file too big\n");
        return 1;
    }

    /* we have the whole config file in memory.  let's parse it ... */
    node = yajl_tree_parse((const char *) fileData, errbuf, sizeof(errbuf));

    /* parse error handling */
    if (node == NULL) {
        fprintf(stderr, "parse_error: ");
        if (strlen(errbuf)) fprintf(stderr, " %s", errbuf);
        else fprintf(stderr, "unknown error");
        fprintf(stderr, "\n");
        return 1;
    }

    /* ... and extract a nested value from the config file */
    {
        //const char * path[] = { "Logging", "timeFormat", (const char *) 0 };

注: "DistServer" を取得しようとすると問題なく動作しますが、"BuildType" は NULL を返します。

        const char * path[] = { "BuildType", (const char *) 0 };
        //const char * path[] = { "DistServer", (const char *) 0 };
        yajl_val v = yajl_tree_get(node, path, yajl_t_string);
        if (v) printf("%s: %s\n", path[0], YAJL_GET_STRING(v));
        else   printf("no such node: %s\n", path[0] );
        //if (v) printf("%s/%s: %s\n", path[0], path[1], YAJL_GET_STRING(v));
        //else   printf("no such node: %s/%s\n", path[0], path[1]);
    }

    yajl_tree_free(node);

    return 0;
}

最新の YAJL バージョン 2.0.2 を使用しています。

前もって感謝します!

編集:

これが私のコマンド出力です:

./parse_config  < ../../example/sample.config 
no such node: BuildType

私が実行している parse_config は build/example ディレクトリにあることに注意してください

私のgccバージョンは次のとおりです:gccバージョン4.4.5(Ubuntu / Linaro 4.4.4-14ubuntu5)

編集2 記録のために、これは付属のサンプルコードです。このコードはYAJLに付属しています。問題が自分のアプリケーションだけに関連していないことを確認するために、自分のコードの代わりに意図的にそれを使用しました。その間、ライブラリが提供するコールバックメカニズムを使用して問題を回避しました

yajl_parse()
and 
yajl_complete_parse()

しかし、元のコードが機能しなかった理由を知りたいです。

4

2 に答える 2

5

バージョン 2.0.2 を使用して、これと同じバグに遭遇しました。これは yajl のバグのようで、github で修正されています (「yajl_tree_get で正しいオブジェクトの長さを検証する」、commit 9c2948a33165c650122d131f31140c15321908f5)。

このパッチを適用したところ、最初の項目を正常に読み取ることができるようになりました。

于 2011-09-10T17:54:33.673 に答える
1

これはyajlのバグのようです...

うわー、そうはならなかったと思いました。

于 2011-09-10T16:23:46.523 に答える