65char *option[N_OPTIONS] = {
76sddsselect [<input1>] <input2> [<output>]\n\
77 [-pipe[=input][,output]]\n\
78 [-match=<column-name>[=<column-name>]]\n\
79 [-equate=<column-name>[=<column-name>]] \n\
81 [-reuse[=rows][,page]] \n\
82 [-majorOrder=row|column]\n\
85 -pipe[=input][,output] Use pipe for input and/or output.\n\
86 -match=<column1>[=<column2>] Specify columns to match between input1 and input2.\n\
87 -equate=<column1>[=<column2>] Specify columns to equate between input1 and input2.\n\
88 -invert Invert the selection to keep non-matching rows.\n\
89 -reuse[=rows][,page] Allow reusing rows or specify page reuse.\n\
90 -majorOrder=row|column Set the output file to row or column major order.\n\
91 -nowarnings Suppress warning messages.\n\
94 sddsselect -match=colA=colB input1.sdds input2.sdds output.sdds\n\
96Program by Michael Borland. (" __DATE__
" " __TIME__
", SVN revision: " SVN_VERSION
")\n";
98int main(
int argc,
char **argv) {
100 long i, j, i_arg, reuse, reusePage;
101 int64_t rows1, rows2, i1, i2;
104 char **match_column, **equate_column;
105 long match_columns, equate_columns;
106 char *input1, *input2, *output;
107 long tmpfile_used, retval1, retval2;
108 long warnings, invert;
109 unsigned long pipeFlags, majorOrderFlag;
110 KEYED_EQUIVALENT **keyGroup = NULL;
112 short columnMajorOrder = -1;
115 argc =
scanargs(&s_arg, argc, argv);
119 input1 = input2 = output = NULL;
120 match_column = equate_column = NULL;
121 match_columns = equate_columns = reuse = reusePage = 0;
122 tmpfile_used = invert = 0;
126 for (i_arg = 1; i_arg < argc; i_arg++) {
127 if (s_arg[i_arg].arg_type == OPTION) {
129 switch (
match_string(s_arg[i_arg].list[0], option, N_OPTIONS, 0)) {
130 case SET_MAJOR_ORDER:
132 s_arg[i_arg].n_items--;
133 if (s_arg[i_arg].n_items > 0 &&
134 (!
scanItemList(&majorOrderFlag, s_arg[i_arg].list + 1, &s_arg[i_arg].n_items, 0,
135 "row", -1, NULL, 0, SDDS_ROW_MAJOR_ORDER,
136 "column", -1, NULL, 0, SDDS_COLUMN_MAJOR_ORDER, NULL)))
137 SDDS_Bomb(
"invalid -majorOrder syntax/values");
138 if (majorOrderFlag & SDDS_COLUMN_MAJOR_ORDER)
139 columnMajorOrder = 1;
140 else if (majorOrderFlag & SDDS_ROW_MAJOR_ORDER)
141 columnMajorOrder = 0;
143 case SET_MATCH_COLUMN:
144 if (s_arg[i_arg].n_items != 2)
146 if (match_columns != 0)
147 SDDS_Bomb(
"only one -match option may be given");
148 match_column =
tmalloc(
sizeof(*match_column) * 2);
149 if ((ptr = strchr(s_arg[i_arg].list[1],
'=')))
152 ptr = s_arg[i_arg].list[1];
153 match_column[0] = s_arg[i_arg].list[1];
154 match_column[1] = ptr;
157 case SET_EQUATE_COLUMN:
158 if (s_arg[i_arg].n_items != 2)
160 if (equate_columns != 0)
161 SDDS_Bomb(
"only one -equate option may be given");
162 equate_column =
tmalloc(
sizeof(*equate_column) * 2);
163 if ((ptr = strchr(s_arg[i_arg].list[1],
'=')))
166 ptr = s_arg[i_arg].list[1];
167 equate_column[0] = s_arg[i_arg].list[1];
168 equate_column[1] = ptr;
172 if (s_arg[i_arg].n_items == 1)
175 char *reuseOptions[2] = {
"rows",
"page"};
176 for (i = 1; i < s_arg[i_arg].n_items; i++) {
177 switch (
match_string(s_arg[i_arg].list[i], reuseOptions, 2, 0)) {
198 if (!
processPipeOption(s_arg[i_arg].list + 1, s_arg[i_arg].n_items - 1, &pipeFlags))
202 fprintf(stderr,
"error: unknown switch: %s\n", s_arg[i_arg].list[0]);
208 input1 = s_arg[i_arg].list[0];
209 else if (input2 == NULL)
210 input2 = s_arg[i_arg].list[0];
211 else if (output == NULL)
212 output = s_arg[i_arg].list[0];
218 if (pipeFlags & USE_STDIN && input1) {
220 SDDS_Bomb(
"too many filenames (sddsxref)");
225 processFilenames(
"sddsselect", &input1, &output, pipeFlags, !warnings, &tmpfile_used);
227 SDDS_Bomb(
"second input file not specified (sddsxref)");
229 if (equate_columns && match_columns)
230 SDDS_Bomb(
"only one of -equate or -match may be given");
231 if (!equate_columns && !match_columns)
232 SDDS_Bomb(
"one of -equate or -match must be given");
246 sprintf(s,
"error: column %s not found or not string type in file %s", match_column[0], input1 ? input1 :
"stdin");
252 sprintf(s,
"error: column %s not found or not string type in file %s", match_column[1], input2);
257 if (equate_columns) {
260 sprintf(s,
"error: column %s not found or not numeric type in file %s", equate_column[0], input1 ? input1 :
"stdin");
266 sprintf(s,
"error: column %s not found or not numeric type in file %s", equate_column[1], input2);
272 if (output && pipeFlags & USE_STDOUT)
273 SDDS_Bomb(
"too many filenames with -pipe option");
274 if (!output && !(pipeFlags & USE_STDOUT)) {
276 fprintf(stderr,
"warning: existing file %s will be replaced (sddsselect)\n", input1 ? input1 :
"stdin");
284 if (columnMajorOrder != -1)
285 SDDS_output.layout.data_mode.column_major = columnMajorOrder;
287 SDDS_output.layout.data_mode.column_major = SDDS_1.layout.data_mode.column_major;
295 fprintf(stderr,
"warning: <input2> ends before <input1>\n");
320 SDDS_SetError(
"Problem copying parameter or array data from second input file");
325 SDDS_SetError(
"Problem copying parameter or array data from first input file");
330 char **string1, **string2;
334 fprintf(stderr,
"Error: problem getting column %s from file %s\n", match_column[0], input1 ? input1 :
"stdin");
337 if (rows2 && !(string2 =
SDDS_GetColumn(&SDDS_2, match_column[1]))) {
338 fprintf(stderr,
"Error: problem getting column %s from file %s\n", match_column[1], input2);
343 for (i1 = 0; i1 < rows1; i1++) {
345 sprintf(s,
"Problem copying row %" PRId64
" of first data set", i1);
353 if ((!matched && !invert) || (matched && invert)) {
359 for (i = 0; i < rows1; i++)
365 for (i = 0; i < rows2; i++)
370 for (i = 0; i < keyGroups; i++) {
372 if (keyGroup[i]->equivalent)
373 free(keyGroup[i]->equivalent);
383 }
else if (equate_columns) {
384 double *value1, *value2;
388 fprintf(stderr,
"Error: problem getting column %s from file %s\n", equate_column[0], input1 ? input1 :
"stdin");
392 fprintf(stderr,
"Error: problem getting column %s from file %s\n", equate_column[1], input2);
397 for (i1 = 0; i1 < rows1; i1++) {
399 sprintf(s,
"Problem copying row %" PRId64
" of first data set", i1);
408 if ((!equated && !invert) || (equated && invert)) {
419 for (i = 0; i < keyGroups; i++) {
421 if (keyGroup[i]->equivalent)
422 free(keyGroup[i]->equivalent);
SDDS (Self Describing Data Set) Data Types Definitions and Function Prototypes.
int32_t SDDS_CopyParameters(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source)
int32_t SDDS_InitializeCopy(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source, char *filename, char *filemode)
int32_t SDDS_CopyRowDirect(SDDS_DATASET *SDDS_target, int64_t target_row, SDDS_DATASET *SDDS_source, int64_t source_row)
int32_t SDDS_CopyPage(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source)
int32_t SDDS_CopyArrays(SDDS_DATASET *SDDS_target, SDDS_DATASET *SDDS_source)
int32_t SDDS_StartPage(SDDS_DATASET *SDDS_dataset, int64_t expected_n_rows)
int32_t SDDS_WritePage(SDDS_DATASET *SDDS_dataset)
Writes the current data table to the output file.
int32_t SDDS_WriteLayout(SDDS_DATASET *SDDS_dataset)
Writes the SDDS layout header to the output file.
void SDDS_SetError(char *error_text)
Records an error message in the SDDS error stack.
int32_t SDDS_GetColumnIndex(SDDS_DATASET *SDDS_dataset, char *name)
Retrieves the index of a named column in the SDDS dataset.
void SDDS_PrintErrors(FILE *fp, int32_t mode)
Prints recorded error messages to a specified file stream.
void SDDS_RegisterProgramName(const char *name)
Registers the executable program name for use in error messages.
int32_t SDDS_GetColumnType(SDDS_DATASET *SDDS_dataset, int32_t index)
Retrieves the data type of a column in the SDDS dataset by its index.
void SDDS_Bomb(char *message)
Terminates the program after printing an error message and recorded errors.
#define SDDS_STRING
Identifier for the string data type.
#define SDDS_DOUBLE
Identifier for the double data type.
#define SDDS_NUMERIC_TYPE(type)
Checks if the given type identifier corresponds to any numeric type.
void * tmalloc(uint64_t size_of_block)
Allocates a memory block of the specified size with zero initialization.
void bomb(char *error, char *usage)
Reports error messages to the terminal and aborts the program.
char * cp_str(char **s, char *t)
Copies a string, allocating memory for storage.
char * delete_chars(char *s, char *t)
Removes all occurrences of characters found in string t from string s.
long match_string(char *string, char **option, long n_options, long mode)
Matches a given string against an array of option strings based on specified modes.
long replaceFileAndBackUp(char *file, char *replacement)
Replaces a file with a replacement file and creates a backup of the original.
int scanargs(SCANNED_ARG **scanned, int argc, char **argv)
long processPipeOption(char **item, long items, unsigned long *flags)
void processFilenames(char *programName, char **input, char **output, unsigned long pipeFlags, long noWarnings, long *tmpOutputUsed)
void free_scanargs(SCANNED_ARG **scanned, int argc)
long scanItemList(unsigned long *flags, char **item, long *items, unsigned long mode,...)
Scans a list of items and assigns values based on provided keywords and types.
KEYED_EQUIVALENT ** MakeSortedKeyGroups(long *keyGroups, long keyType, void *data, long points)
Create sorted key groups from data.
long FindMatchingKeyGroup(KEYED_EQUIVALENT **keyGroup, long keyGroups, long keyType, void *searchKeyData, long reuse)
Find a matching key group for a search key.
char * tmpname(char *s)
Supplies a unique temporary filename.