Change license from GPLv2+ to GPLv3+.
[pspp-builds.git] / src / language / syntax-file.c
1 /* PSPP - a program for statistical analysis.
2    Copyright (C) 1997-9, 2000 Free Software Foundation, Inc.
3
4    This program is free software: you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation, either version 3 of the License, or
7    (at your option) any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program.  If not, see <http://www.gnu.org/licenses/>. */
16
17 #include <config.h>
18
19 #include "syntax-file.h"
20
21 #include <stdio.h>
22 #include <errno.h>
23 #include <stdlib.h>
24
25 #include <data/file-name.h>
26 #include <data/settings.h>
27 #include <data/variable.h>
28 #include <language/command.h>
29 #include <language/lexer/lexer.h>
30 #include <libpspp/alloc.h>
31 #include <libpspp/assertion.h>
32 #include <libpspp/message.h>
33 #include <libpspp/message.h>
34 #include <libpspp/str.h>
35 #include <libpspp/verbose-msg.h>
36 #include <libpspp/version.h>
37 #include <output/table.h>
38
39 #include <libpspp/ll.h>
40
41 #include "prompt.h"
42
43 #include "gettext.h"
44 #define _(msgid) gettext (msgid)
45
46 #include <libpspp/getl.h>
47
48
49 struct syntax_file_source
50   {
51     struct getl_interface parent ;
52
53     FILE *syntax_file;
54
55     /* Current location. */
56     char *fn;                           /* File name. */
57     int ln;                             /* Line number. */
58   };
59
60 static const char *
61 name (const struct getl_interface *s)
62 {
63   const struct syntax_file_source *sfs = (const struct syntax_file_source *) s;
64   return sfs->fn;
65 }
66
67 static int
68 line_number (const struct getl_interface *s)
69 {
70   const struct syntax_file_source *sfs = (const struct syntax_file_source *) s;
71   return sfs->ln;
72 }
73
74
75 /* Reads a line from syntax file source S into LINE.
76    Returns true if successful, false at end of file. */
77 bool
78 read_syntax_file (struct getl_interface *s,
79                   struct string *line, enum getl_syntax *syntax)
80 {
81   struct syntax_file_source *sfs = (struct syntax_file_source *) s;
82
83   /* Open file, if not yet opened. */
84   if (sfs->syntax_file == NULL)
85     {
86       verbose_msg (1, _("opening \"%s\" as syntax file"), sfs->fn);
87       sfs->syntax_file = fn_open (sfs->fn, "r");
88
89       if (sfs->syntax_file == NULL)
90         {
91           msg (ME, _("Opening `%s': %s."), sfs->fn, strerror (errno));
92           return false;
93         }
94     }
95
96   /* Read line from file and remove new-line.
97      Skip initial "#! /usr/bin/pspp" line. */
98   do
99     {
100       sfs->ln++;
101       if (!ds_read_line (line, sfs->syntax_file))
102         {
103           if (ferror (sfs->syntax_file))
104             msg (ME, _("Reading `%s': %s."), sfs->fn, strerror (errno));
105           return false;
106         }
107       ds_chomp (line, '\n');
108     }
109   while (sfs->ln == 1 && !memcmp (ds_cstr (line), "#!", 2));
110
111   /* Echo to listing file, if configured to do so. */
112   if (get_echo ())
113     tab_output_text (TAB_LEFT | TAB_FIX, ds_cstr (line));
114
115   *syntax = GETL_BATCH;
116   return true;
117 }
118
119 static void
120 syntax_close (struct getl_interface *s)
121 {
122   struct syntax_file_source *sfs = (struct syntax_file_source *) s;
123
124   if (sfs->syntax_file && EOF == fn_close (sfs->fn, sfs->syntax_file))
125     msg (MW, _("Closing `%s': %s."), sfs->fn, strerror (errno));
126   free (sfs->fn);
127   free (sfs);
128 }
129
130 static bool
131 always_false (const struct getl_interface *s UNUSED)
132 {
133   return false;
134 }
135
136
137 /* Creates a syntax file source with file name FN. */
138 struct getl_interface *
139 create_syntax_file_source (const char *fn)
140 {
141   struct syntax_file_source *ss = xzalloc (sizeof (*ss));
142
143   ss->fn = xstrdup (fn);
144
145   ss->parent.interactive = always_false;
146   ss->parent.read = read_syntax_file ;
147   ss->parent.filter = NULL;
148   ss->parent.close = syntax_close ;
149   ss->parent.name = name ;
150   ss->parent.location = line_number;
151
152   return (struct getl_interface *) ss;
153 }
154