-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathCsvGrailsPlugin.groovy
185 lines (151 loc) · 7.2 KB
/
CsvGrailsPlugin.groovy
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
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
/*
* Copyright 2010 Les Hazlewood
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* under the License.
*/
import org.grails.plugins.csv.CSVReaderUtils
import org.grails.plugins.csv.CSVMapReader
import au.com.bytecode.opencsv.CSVReader
import org.grails.plugins.csv.controller.RenderCsvMethod
class CsvGrailsPlugin {
// the plugin version
def version = "0.3"
// the version or versions of Grails the plugin is designed for
def grailsVersion = "1.2 > *"
// the other plugins this plugin depends on
def dependsOn = [:]
// resources that are excluded from plugin packaging
def pluginExcludes = [
"grails-app/views/error.gsp",
"grails-app/controllers/**/*"
]
def observe = [
"controllers"
]
def loadAfter = [
"controllers"
]
// TODO Fill in these fields
def author = "Les Hazlewood"
def authorEmail = "[email protected]"
def title = "Grails CSV Plugin"
def description = '''
The Grails CSV Plugin allows you to easily parse and consume CSV data from a number of input sources. It
supports complex parsing scenarios such as nested commas inside quotes, escaped tokens, multi-line quoted
values and allows configuration of parsing options (separator char, escape char, text encoding, etc). It is
based on Glen Smith (et. al.)'s OpenCSV project (http://opencsv.sourceforge.net)
This plugin adds two dynamic methods 'eachCsvLine' and 'toCsvReader' to each of the following classes:
- java.lang.String
- java.io.File
- java.io.InputStream
- java.io.Reader
Using it is extremely simple. On any instance of the four data types, call the 'eachCvsLine' method with a
closure accepting the tokens (a String array) for each parsed line:
"hello, world, how, are, you".eachCsvLine { tokens ->
//only one line in this case and tokens.length == 5
}
new File("iso3166Countries.csv").eachCsvLine { tokens ->
new Country(tokens[0], //ISO 3166 country name
tokens[1]).save() //ISO 3166 2 letter character code
}
Configuration
If you need to specify how the parsing should occur, you can construct your own csv reader with a map of
configuration options and call the 'eachLine' method on the constructed reader:
anInputStream.toCsvReader(['charset':'UTF-8']).eachLine { tokens ->
...
}
The supported config options:
'separatorChar': the character to use as the delimiter to separate the tokens. Defaults to the comma: ','
'quoteChar': the character indicating a quoted string is about to follow. Internal separatorChars can be
inside the quoted string and they will not be split into tokens.
Defaults to the double quote char: '"'
'escapeChar': the character to escape an immediately following character, indicating to the parser not to treat
it as a special char. Defaults to the backslash char: '\'
'skipLines': the number of lines in the input source to skip before parsing begins. This is useful to skip
any potential CSV header lines that are not part of the CSV data. Defaults to zero '0'
'strictQuotes': if characters outside of quotes should be ignored (implying each individual token is
quoted. Defaults to false
'ignoreLeadingWhiteSpace': white space in front of a quoted token is ignored. Defaults to true
'charset': use the specified charset when parsing an InputStream. The value can be either the Charset name
as a String, a java.nio.charset.Charset instance, or a java.nio.charset.CharsetDecoder instance.
Defaults to the system default charset.
*Note that this option is ONLY valid for InputStream instances. It is ignored otherwise.
'''
// URL to the plugin's documentation
def documentation = "http://grails.org/plugin/csv"
def doWithWebDescriptor = { xml ->
// TODO Implement additions to web.xml (optional), this event occurs before
}
def doWithSpring = {
// TODO Implement runtime spring config (optional)
}
def renderCsvMethod = { Map args, Closure definition ->
new RenderCsvMethod(delegate).call(args, definition)
false
}
def doWithDynamicMethods = { ctx ->
//TODO this should really all be in a utility method so it can easily be called for unit testing
CSVReader.metaClass.eachLine = { closure ->
CSVReaderUtils.eachLine((CSVReader) delegate, closure)
}
File.metaClass.eachCsvLine = { closure ->
CSVReaderUtils.eachLine((File) delegate, closure)
}
File.metaClass.toCsvReader = { settingsMap ->
return CSVReaderUtils.toCsvReader((File)delegate, settingsMap)
}
File.metaClass.toCsvMapReader = { settingsMap ->
return new CSVMapReader(new FileReader(delegate),settingsMap)
}
InputStream.metaClass.eachCsvLine = { closure ->
CSVReaderUtils.eachLine((InputStream) delegate, closure)
}
InputStream.metaClass.toCsvReader = { settingsMap ->
return CSVReaderUtils.toCsvReader((InputStream)delegate, settingsMap)
}
Reader.metaClass.eachCsvLine = { closure ->
CSVReaderUtils.eachLine((Reader)delegate, closure)
}
Reader.metaClass.toCsvReader = { settingsMap ->
return CSVReaderUtils.toCsvReader((Reader)delegate, settingsMap)
}
Reader.metaClass.toCsvMapReader = { settingsMap ->
return new CSVMapReader(delegate,settingsMap)
}
String.metaClass.eachCsvLine = { closure ->
CSVReaderUtils.eachLine((String) delegate, closure)
}
String.metaClass.toCsvReader = { settingsMap ->
return CSVReaderUtils.toCsvReader((String)delegate, settingsMap)
}
String.metaClass.toCsvMapReader = { settingsMap ->
return new CSVMapReader(new StringReader(delegate),settingsMap)
}
application.controllerClasses.each {
it.clazz.metaClass.renderCsv = renderCsvMethod
}
}
def doWithApplicationContext = { applicationContext ->
// TODO Implement post initialization spring config (optional)
}
def onChange = { event ->
if (application.isControllerClass(event.source)) {
event.source.metaClass.renderCsv = renderCsvMethod
}
}
def onConfigChange = { event ->
// TODO Implement code that is executed when the project configuration changes.
// The event is the same as for 'onChange'.
}
}