-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathv_colaspds.pro
528 lines (462 loc) · 19.1 KB
/
v_colaspds.pro
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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
function V_COLASPDS, input_name, labelIn, Odef, KEYWDS = keywds, INFO = info, $
SILENT = silent
;+ $Id: v_colaspds.pro,v 1.9 2013/11/05 13:22:08 sophie Exp $
;
; NAME:
; V_COLASPDS (collection - ascii - pds)
; PURPOSE:
; Read a PDS ascii collection into IDL structure containing the elements
; of the collection as elements.
;
; CALLING SEQUENCE:
; Result=V_COLASPDS ( Input_name, label [,KEYWDS = keywds, INFO=info, $
; /SILENT] )
;
; INPUTS:
; INPUT_NAME = Either a scalar string containing the name of the PDS file
; to be read or a byte array to be converted into the appropriate
; PDS collection. The latter input would normally be used only
; when V_COLASPDS is called by ARASCPDS to read a collection array
; object or recursively by itself to read a collection collection
; object.
;
; LABEL = String array containing the "header" from the PDS file.
; Required if and only if the input type is a filename.
;
; Odef = "pointer" from v_objpds defining the object to be read
;
; OUTPUTS:
; Result = PDS IDL structure constructed from designated input.
;
; OPTIONAL INPUT KEYWORDS:
;
; SILENT - Normally, V_COLASPDS will display the size of the array at
; the terminal. The SILENT keyword will suppress this
;
;
; INFO - A string array giving the names and sizes of all arrays
; contained in the output, with number of elements = the number
; of arrays created. Especially useful in the case of nested
; arrays (array of arrays), see example in header of
; ARASCPDS.PRO. A nested bracket notation is used, see same
; example. You should rename this variable when doing
; multiple reads as it is passed as both input an output and any
; new info will be concatanated on top of the old info.
;
; OPTIONAL OUTPUT KEYWORDS:
;
; KEYWDS - Required if and only if the input type is a byte array.
; Supplies the label keywords which tells V_COLASPDS.PRO how to
; interpret the data array. It is an nx8-element string array
; which must conform to a (8,n), i.e. transposed, format, where
; n = the total number of objects in the sub-collection being
; extracted (including the coolection itself as the first
; object). The 8 columns must contain the following keywords
; from the PDS label:
; 1. obJECT
; 2. the index of the 'ENDOBJECT' corrsponding to each
; OBJECT. Note that it will NOT be monotonically
; increasing since there are embedded OBJECTS, e.g.
; the 0th index will always be the highest number since
; the 0th OBJECT will be the collection itself.
; 3. The START_BYTE corrsponding to OBJECT.
; 4. The NAME of the OBJECT.
; 5. BYTES used by the OBJECT. If the OBJECT has no BYTES
; keyword, set to -1.
; 6. DATA_TYPE of the OBJECT. If the OBJECT has no DATA_TYPE
; keyword, set to ' ' (whitespace).
; 7. AXES if the OBJECT is an array. IF not an array, set = 0.
; 8. AXIS_ITEMS if the OBJECT is an array. IF not an array, set
; to ' ' (whitespace).
; Note that all numerical values must be converted to string
; before concatanating the KEYWDS array.
;
; EXAMPLE:
; Read a PDS collection file ASCOL.PDS into an IDL structure whose
; elements are the elemenst of the collection, and get info on any
; arrays which are contained in the collection:
;
; IDL> col = V_COLASPDS( 'ASCOL.PDS', lbl, info=info)
;
; If you want to read only a sub-collection (an element of the collection
; which is itself a collection), and get info on arrays which are
; contained in the sub-collection:
;
; IDL> col = V_COLASPDS( test_data,keywds=keywds,info=info)
; where test_data is a byte array extracted from the data read from
; the file ASCOL.PDS. The extraction goes from the START_BYTE of
; the sub-collection being extracted to the end of the data.
;
; PROCEDURES USED:
; Functions: V_PDSPAR, V_STR2NUM
;
; MODIFICATION HISTORY:
; Written by Michael E. Haken, May-June 1996
; Portions adapted from TASCPDS by John D. Koch
; Modified for VIRTIS from SBNIDL 2.0, Stephane Erard, sept. 2000
; Fixed search for object pointer
; Stephane Erard, sept. 2012:
; Now uses an extra input argument (object definition structure) and
; restrains label search to the part related to object under consideration
; The argument passing system is a mess, but this is required to
; have v_arascpds working.
; Changed file opening mechanism.
;
;-
;
;###########################################################################
;
; LICENSE
;
; Copyright (c) 1999-2008, StŽphane Erard, CNRS - Observatoire de Paris
; All rights reserved.
; Non-profit redistribution and non-profit use in source and binary forms,
; with or without modification, are permitted provided that the following
; conditions are met:
;
; Redistributions of source code must retain the above copyright
; notice, this list of conditions, the following disclaimer, and
; all the modifications history.
; Redistributions in binary form must reproduce the above copyright
; notice, this list of conditions, the following disclaimer and all the
; modifications history in the documentation and/or other materials
; provided with the distribution.
; Neither the name of the CNRS and Observatoire de Paris nor the
; names of its contributors may be used to endorse or promote products
; derived from this software without specific prior written permission.
;
; THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND ANY
; EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
; WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
; DISCLAIMED. IN NO EVENT SHALL THE REGENTS AND CONTRIBUTORS BE LIABLE FOR ANY
; DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
; (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
; ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
; (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
; SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
;
On_error,2 ;Return to user
; Check for proper input
if keyword_set(keywds) then begin
if N_params() LT 1 then begin
print,'Syntax - result = V_COLASPDS( input_name, KEYWDS=keywds '+ $
'[,INFO=info ,/SILENT])'
return, -1
endif
endif else begin
if N_params() LT 2 then begin
print,'Syntax - result = V_COLASPDS( input_name, lbl[,INFO=info ,/SILENT])'
return, -1
endif
endelse
silent = keyword_set( SILENT )
fname = input_name
sinput=size(fname)
type_input=sinput(sinput(0)+1)
if (type_input ne 1 and type_input ne 7) or $
(type_input eq 7 and sinput(0) ne 0) then message, $
'Input must be either scalar string (filename) or byte array'
if type_input eq 1 then begin
if not (keyword_set(keywds)) then message, $
'Must input KEYWDS if input is a byte array' $
else keywds_tmp=keywds
endif
; Checkout array pointer (?)
if (Odef.pointer EQ '') then begin
if not (SILENT) then message, /cont, 'ERROR - No pointer to array data found in '+Odef.name
return, -1
endif
; restrain label to useful part
label = labelIn[Odef.start:Odef.stop]
record_bytes = long(v_pdspar(labelIn,'RECORD_BYTES'))
txt=fname
addtxt=''
if type_input eq 7 then begin
; Read object to determine type of data in file
object = v_pdspar(label,'OBJECT',COUNT=objects,INDEX=obj_ind)
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+': missing required OBJECT keywords'
object=strlowcase(object)
inform = v_pdspar( label, 'INTERCHANGE_FORMAT', COUNT=informcount )
if !ERR EQ -1 then begin
message,'ERROR - '+txt+': missing required INTERCHANGE_FORMAT keyword'
endif else begin
binform=byte(inform)
exchr=where(binform eq 39b or binform eq 34b) ; remove characters
if exchr(0) ne -1 then binform(exchr)=32b ;"(34b) and '(39b)
inform=strtrim(binform,2) ; from inform
if n_elements(inform) gt 1 then begin
message, txt+': '+strcompress(informcount)+' values found for ' +$
'INTERCHANGE_FORMAT keyword. Attempting to proceed.', /INFORM
if n_elements(uniq(inform)) gt 1 then message, $
'ERROR - '+txt+': conflicting values of INTERCHANGE_FORMAT keyword.'
endif
inform = inform(0)
if inform EQ "BINARY" then message, $
'ERROR- '+txt+' is an binary collection file; try COLBIPDS.'
endelse
name = v_pdspar(label,'NAME', COUNT= ncount,INDEX=nam_ind)
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required NAME keywords'
data_type = v_pdspar(label,'DATA_TYPE',COUNT= dcount,INDEX=typ_ind)
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required DATA_TYPE keywords'
length = v_pdspar(label,'BYTES',COUNT=bcount,INDEX=len_ind)
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required BYTES keywords'
start_byte = v_pdspar(label,'START_BYTE',COUNT=starts,INDEX=st_ind) - 1
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required START_BYTE keywords'
end_object=v_pdspar(label,'END_OBJECT',COUNT=end_objects,INDEX=eobj_ind)
endobj_ind=where(strpos(label,'END_OBJECT') ne -1,endobjects)
if objects ne endobjects then message, $
txt+addtxt+ $
': discrepancy in number of OBJECT keywords/END_OBJECT keywords. '+$
'Attempting to proceed.',/INFORM
if total(strpos(object,'ARRAY')) gt -1*objects then begin
axes = fix(v_pdspar(label,'AXES',COUNT=axcount,INDEX=axes_ind))
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required AXES keyword'
axis_items = v_pdspar(label,'AXIS_ITEMS',COUNT=axitcount,INDEX=axit_ind)
if !ERR EQ -1 then message, $
'ERROR - '+txt+addtxt+' missing required AXIS_ITEMS keyword'
if axcount ne axitcount then message, $
'ERROR - '+txt+' discrepency in number of AXES keywords/AXIS_ITEMS keywords'
endif else axcount=0
start_byte_tmp=start_byte
name_tmp=name
len_tmp=length
typ_tmp=data_type
if n_elements(axes) gt 0 then begin
axes_tmp=axes
axis_items_tmp=axis_items
endif
start_byte=lonarr(objects)
name=strarr(objects)+' '
length=lonarr(objects)-1
data_type=strarr(objects)+' '
axes=lonarr(objects)
axis_items=strarr(objects)+' '
for i=0,objects-1 do begin
ind_lo = obj_ind(i)
if i lt objects -1 then $
ind_hi = obj_ind(i+1) else ind_hi = n_elements(label)
for j=0,starts-1 do $
if st_ind(j) gt ind_lo and st_ind(j) lt ind_hi then $
start_byte(i)=start_byte_tmp(j)
for j=0,ncount-1 do $
if nam_ind(j) gt ind_lo and nam_ind(j) lt ind_hi then $
name(i)=name_tmp(j)
for j=0,bcount-1 do $
if len_ind(j) gt ind_lo and len_ind(j) lt ind_hi then $
length(i)=len_tmp(j)
for j=0,dcount-1 do $
if typ_ind(j) gt ind_lo and typ_ind(j) lt ind_hi then $
data_type(i)=typ_tmp(j)
if axcount gt 0 then begin
for j=0,axcount-1 do begin
if axes_ind(j) gt ind_lo and axes_ind(j) lt ind_hi then $
axes(i)=axes_tmp(j)
if axit_ind(j) gt ind_lo and axit_ind(j) lt ind_hi then $
axis_items(i)=axis_items_tmp(j)
endfor
endif
endfor
; Trim extraneous characters from column names and data_types
bname=byte(name)
exchr=where(bname eq 10b or bname eq 13b or bname eq 34b $
or bname eq 39b or bname eq 40b or bname eq 41b)
if exchr(0) ne -1 then bname(exchr)=32b
bdata_type=byte(data_type)
exchr=where(bdata_type eq 10b or bdata_type eq 13b or bdata_type eq 34b $
or bdata_type eq 39b or bdata_type eq 40b or bdata_type eq 41b)
if exchr(0) ne -1 then bdata_type(exchr)=32b
name = strtrim(bname,2)
data_type = strtrim(bdata_type,2)
; Read pointer to find location of the collection data
CASE !version.os_family OF
'Windows' : sep= '\'
'unix' : sep = '/'
'MacOS' : sep = ':'
ELSE: sep='' ; assumes coordinates in the same directory
ENDCASE
; pointer = v_pdspar(label,'^COLLECTION')
; if !ERR EQ -1 then message, $
; 'ERROR- '+txt+addtxt+': missing valid file pointer'
; point = pointer(0)
; skip=0L
; temp = v_str2num(point,TYPE=t)
; if t GT 6 then begin
; l = strlen(point)
; p = strpos(point,'"')
; p2 = strpos(point,'"',p+1)
; if p LT 0 then begin
; p = strpos(point,"'")
; p2 = strpos(point,"'",p+1)
; endif
; c = strpos(point,',')
; if (c GT -1) then skip = long(v_str2num(strtrim(strmid(point,c+1,L-c-1),2))-1)
; if (p GT -1) then point=strmid(point,p+1,p2-p-1)
; d = strpos(fname,sep)
; tail = d
; dir = ''
; while d GT -1 do begin ; extract the path to the directory,
; tail = d + 1
; d = strpos(fname,sep,tail)
; endwhile
; dir = strmid(fname,0,tail)
; fname = dir + strupcase(point)
; openr, unit, fname, ERROR = err, /GET_LUN, /BLOCK
; if err LT 0 then fname = dir + strlowcase(point)
; openr, unit, fname, ERROR = err, /GET_LUN, /BLOCK
; if err LT 0 then begin
; fname = dir + (point)
; openr, unit, fname, ERROR = err, /GET_LUN, /BLOCK
; endif
; if err LT 0 then message,'Error opening file ' + ' ' + fname
; endif else begin
; skip = temp -1
; endelse
PtObj = V_POINTPDS(Odef.pointer,record_bytes)
datafile_found = (PtObj.filename NE '')
if datafile_found NE 0 then begin ; if detached label, look for file location
fname = file_search(PtObj.filename, /fold) ; works from IDL 5.5 and up
temp = file_info(fname)
; If not found in current directory, try in label directory
if fname EQ "" or not(temp.exists) then begin
DirName = v_getpath(input_name, FBname) ; get path to label under IDL ³ 5.4
fname = file_search(Dirname+PtObj.filename, /fold)
temp = file_info(fname)
endif
if fname EQ "" or not(temp.exists) then message, 'ERROR - Could not re-open '+ PtObj.filename
endif
openr, unit, fname, ERROR = err, /GET_LUN, /compress
if err LT 0 then message,'Error opening file ' + ' ' + fname
; Inform user of program status if /SILENT not set
if not (SILENT) then $
message,'Now reading collection from ' +fname,/INFORM
; Read data into a 1-dimensional byte array
filestat=fstat(unit)
XY = filestat.size
filedata=bytarr(XY,/NOZERO)
point_lun,unit,skip
readu,unit,filedata
free_lun, unit
cr=where(filedata eq 13b,ctcr)
lf=where(filedata eq 10b,ctlf)
if cr(0) LT 0 then $
print,'ERROR IN DATA FILE: No carriage return characters found.' + $
'Attempting to proceed.'
if lf(0) LT 0 then begin
print, $
'ERROR IN DATA FILE: No line feed characters found. Attempting to proceed.'
endif
if lf(0) GT 0 and cr(0) GT 0 then $
if not (ctcr EQ ctlf and total(lf-cr) EQ ctcr) then begin
print,'ERROR IN DATA FILE: Carriage return + line feed should ' + $
'terminate each line. Attempting to proceed.'
bad_line_term=1
endif
filedata=filedata(where(filedata ne 10b and filedata ne 13b))
eo=obj_ind
e=[[endobj_ind],[replicate(-1,objects)]]
oi=[[obj_ind],[replicate(1,objects)]]
oi=[oi,e]
oi=oi(sort(oi(*,0)),*)
ind=where(oi(*,1) eq 1)
for j=0,objects-1 do begin
t=0
for i=ind(j),n_elements(oi)/2-1 do begin
t=t+oi(i,1)
if t(0) eq 0 then begin
if oi(i,1) eq -1 then eo(j)=oi(i,0)
goto, continue
endif
endfor
continue:
endfor
endif else begin
XY = n_elements(fname)
filedata = fname
start_byte=long(reform(keywds(2,*)))
start_byte(0)=0
object=reform(keywds(0,*))
objects=n_elements(object)
eo=long(reform(keywds(1,*)))
name=reform(keywds(3,*))
length=long(reform(keywds(4,*)))
data_type=reform(keywds(5,*))
axes=long(reform(keywds(6,*)))
axis_items=reform(keywds(7,*))
endelse
endobj_order=sort(sort(eo))
nestend=0
for j=1,endobj_order(0) do begin
if endobj_order(j) lt nestend then goto, nextj
start=start_byte(j)
typ=data_type(j)
bytes=length(j)
if object(j) eq 'element' then begin
vect=string(filedata(start:start+bytes-1))
typ = data_type(j)
if strmid(typ(0),0,5) eq 'ASCII' then $
typ(0)=strmid(typ(0),6,strlen(typ(0))-6)
CASE typ(0) OF
'INTEGER': vect = long(vect)
'UNSIGNED_INTEGER': vect = long(vect)
'REAL': vect = double(vect)
'FLOAT': vect = double(vect)
'CHARACTER': vect=vect
'DOUBLE': vect = double(vect)
'BYTE': vect = long(vect)
'BOOLEAN': vect = long(vect)
'TIME': vect=vect
'DATE': vect=vect
else: message,$
typ(0)+' not a recognized data type!'
ENDCASE
endif
if object(j) eq 'collection' then begin
if bytes eq -1 then message, $
'collection '+name(j)+'missing required BYTES keyword. '+ $
'Attempting to proceed',/INFORM
nestend=endobj_order(j)
eo_order=sort(sort(eo(j:*)))
last=eo_order(0)
keywds=[transpose(object(j:*)),string([transpose(eo(j:*)), $
transpose(start_byte(j:*))]),transpose(name(j:*)),$
string(transpose(length(j:*))),transpose(data_type(j:*)), $
string(transpose(axes(j:*))),transpose(axis_items(j:*))]
keywds=keywds(*,0:last)
vect=v_colaspds(filedata(start:*),keywds=keywds,info=info,silent=silent)
endif
if object(j) eq 'array' then begin
nestend=endobj_order(j)
eo_order=sort(sort(eo(j:*)))
last=eo_order(0)
keywds=[transpose(object(j:*)),string([transpose(eo(j:*)), $
transpose(start_byte(j:*))]),transpose(name(j:*)),$
string(transpose(length(j:*))),transpose(data_type(j:*)), $
string(transpose(axes(j:*))),transpose(axis_items(j:*))]
keywds=keywds(*,0:last)
; vect=v_arascpds(filedata(start:*),keywds=keywds,info=info,silent=silent)
vect=v_arascpds(filedata(start:*),labelIn, Odef,keywds=keywds,info=info,silent=silent)
endif
if j eq 1 then collection=create_struct(name(j),vect) else $
collection=create_struct(collection,name(j),vect)
struct_name = name(0)
nextj:
endfor
collection=create_struct(name=struct_name,collection)
if type_input eq 7 then begin
if not (SILENT) then begin
; print,'Created:'
help, collection,/st
print,' '
endif
endif
; Return data table in IDL array form
if type_input eq 1 then keywds=keywds_tmp
return, collection
end