File: pt_peg_to_json.tcl

package info (click to toggle)
tcllib 1.14-dfsg-3%2Bdeb7u1
  • links: PTS
  • area: main
  • in suites: wheezy
  • size: 33,036 kB
  • sloc: tcl: 148,302; ansic: 14,067; sh: 10,320; xml: 1,766; yacc: 753; pascal: 551; makefile: 129; perl: 84; f90: 84; python: 33; ruby: 13; php: 11
file content (149 lines) | stat: -rw-r--r-- 3,774 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
# peg_to_json.tcl --
#
#	Conversion from PEG to JSON (Java Script Object Notation).
#
# Copyright (c) 2009 Andreas Kupries <andreas_kupries@sourceforge.net>
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
# 
# RCS: @(#) $Id: pt_peg_to_json.tcl,v 1.1 2010/03/26 05:07:24 andreas_kupries Exp $

# This package takes the canonical serialization of a parsing
# expression grammar and produces text in JSON format, Java Script
# data transfer format.

# ### ### ### ######### ######### #########
## Requisites

package require Tcl 8.5
package require pt::peg      ; # Verification that the
					       # input is proper.
package require json::write

# ### ### ### ######### ######### #########
##

namespace eval ::pt::peg::to::json {
    namespace export \
	reset configure convert

    namespace ensemble create
}

# ### ### ### ######### ######### #########
## API.

proc ::pt::peg::to::json::reset {} {
    variable indented 0
    variable aligned  0
    variable name     a_pe_grammar
    variable file     unknown
    variable user     unknown
    return
}

proc ::pt::peg::to::json::configure {args} {
    variable indented
    variable aligned
    variable name
    variable file
    variable user

    if {[llength $args] == 0} {
	return [list \
		    -file     $file \
		    -name     $name \
		    -user     $user \
		    -indented $indented \
		    -aligned  $aligned]
    } elseif {[llength $args] == 1} {
	lassign $args option
	set variable [string range $option 1 end]
	if {[info exists $variable]} {
	    return [set $variable]
	} else {
	    return -code error "Expected one of -aligned, or -indented, got \"$option\""
	}
    } elseif {[llength $args] % 2 == 0} {
	foreach {option value} $args {
	    set variable [string range $option 1 end]
	    if {![info exists $variable]} {
		return -code error "Expected one of -aligned, or -indented, got \"$option\""
	    }
	}
	foreach {option value} $args {
	    set variable [string range $option 1 end]
	    switch -exact -- $variable {
		indented - aligned {
		    if {![::string is boolean -strict $value]} {
			return -code error "Expected boolean, got \"$value\""
		    }
		}
		name -
		file -
		user { }
	    }
	    set $variable $value
	}
    } else {
	return -code error {wrong#args, expected option value ...}
    }
}

proc ::pt::peg::to::json::convert {serial} {
    variable indented
    variable aligned

    ::pt::peg verify-as-canonical $serial

    json::write indented $indented
    json::write aligned  $aligned

    # Unpack the serialization, known as canonical
    array set peg $serial
    array set peg $peg(pt::grammar::peg)
    unset     peg(pt::grammar::peg)

    # Assemble the rules object
    set rules {}
    foreach {symbol def} $peg(rules) {
	lassign $def _ is _ mode
	lappend rules $symbol \
	    [json::write object \
		 is   [json::write string $is] \
		 mode [json::write string $mode]]
    }

    # Assemble the final result
    return [json::write object pt::grammar::peg \
		[json::write object \
		     rules [json::write object {*}$rules] \
		     start [json::write string $peg(start)]]]

    # ### ### ### ######### ######### #########
}

# ### ### ### ######### ######### #########
## Configuration

namespace eval ::pt::peg::to::json {

    # Combinations of the format specific entries
    # I A |
    # - - + ---------------------
    # 0 0 | Ultracompact (no whitespace, single line)
    # 1 0 | Indented
    # 0 1 | Not possible, per the implications above.
    # 1 1 | Indented + Tabular aligned keys
    # - - + ---------------------

    variable indented 0
    variable aligned  0
}

# ### ### ### ######### ######### #########
## Ready

package provide pt::peg::to::json 1
return