regress.c 14.3 KB
Newer Older
1
/*
2
 * $Header: /cvsroot/pgsql/src/test/regress/regress.c,v 1.39 2000/06/11 20:07:44 tgl Exp $
3 4
 */

5
#include <float.h>				/* faked on sunos */
6

7
#include "postgres.h"
M
Marc G. Fournier 已提交
8

9
#include "utils/geo_decls.h"	/* includes <math.h> */
10
#include "executor/executor.h"	/* For GetAttributeByName */
11
#include "commands/sequence.h"	/* for nextval() */
12 13

#define P_MAXDIG 12
14 15 16
#define LDELIM			'('
#define RDELIM			')'
#define DELIM			','
17

18
typedef TupleTableSlot *TUPLE;
19

20 21 22 23 24
extern double *regress_dist_ptpath(Point *pt, PATH *path);
extern double *regress_path_dist(PATH *p1, PATH *p2);
extern PATH *poly2path(POLYGON *poly);
extern Point *interpt_pp(PATH *p1, PATH *p2);
extern void regress_lseg_construct(LSEG *lseg, Point *pt1, Point *pt2);
25
extern Datum overpaid(PG_FUNCTION_ARGS);
B
Bruce Momjian 已提交
26
extern int	boxarea(BOX *box);
27
extern char *reverse_name(char *string);
28 29

/*
30
** Distance from a point to a path
31
*/
32
double *
33
regress_dist_ptpath(pt, path)
34 35
Point	   *pt;
PATH	   *path;
36
{
37 38 39 40
	double	   *result;
	double	   *tmp;
	int			i;
	LSEG		lseg;
41 42 43

	switch (path->npts)
	{
44
		case 0:
45
			result = palloc(sizeof(double));
46 47 48 49 50 51 52 53 54 55 56 57
			*result = Abs((double) DBL_MAX);	/* +infinity */
			break;
		case 1:
			result = point_distance(pt, &path->p[0]);
			break;
		default:

			/*
			 * the distance from a point to a path is the smallest
			 * distance from the point to any of its constituent segments.
			 */
			Assert(path->npts > 1);
58
			result = palloc(sizeof(double));
59 60 61 62 63 64
			for (i = 0; i < path->npts - 1; ++i)
			{
				regress_lseg_construct(&lseg, &path->p[i], &path->p[i + 1]);
				tmp = dist_ps(pt, &lseg);
				if (i == 0 || *tmp < *result)
					*result = *tmp;
65
				pfree(tmp);
66 67 68

			}
			break;
69
	}
70
	return result;
71 72 73 74
}

/* this essentially does a cartesian product of the lsegs in the
   two paths, and finds the min distance between any two lsegs */
75
double *
76
regress_path_dist(p1, p2)
77 78
PATH	   *p1;
PATH	   *p2;
79
{
80 81 82 83 84 85
	double	   *min,
			   *tmp;
	int			i,
				j;
	LSEG		seg1,
				seg2;
86 87 88 89 90 91 92 93 94 95 96 97 98

	regress_lseg_construct(&seg1, &p1->p[0], &p1->p[1]);
	regress_lseg_construct(&seg2, &p2->p[0], &p2->p[1]);
	min = lseg_distance(&seg1, &seg2);

	for (i = 0; i < p1->npts - 1; i++)
		for (j = 0; j < p2->npts - 1; j++)
		{
			regress_lseg_construct(&seg1, &p1->p[i], &p1->p[i + 1]);
			regress_lseg_construct(&seg2, &p2->p[j], &p2->p[j + 1]);

			if (*min < *(tmp = lseg_distance(&seg1, &seg2)))
				*min = *tmp;
99
			pfree(tmp);
100 101
		}

102
	return min;
103 104
}

105
PATH *
106
poly2path(poly)
107
POLYGON    *poly;
108
{
109
	int			i;
110
	char	   *output = (char *) palloc(2 * (P_MAXDIG + 1) * poly->npts + 64);
111
	char		buf[2 * (P_MAXDIG) + 20];
112

113
	sprintf(output, "(1, %*d", P_MAXDIG, poly->npts);
114

115 116 117 118 119
	for (i = 0; i < poly->npts; i++)
	{
		sprintf(buf, ",%*g,%*g", P_MAXDIG, poly->p[i].x, P_MAXDIG, poly->p[i].y);
		strcat(output, buf);
	}
120

121 122
	sprintf(buf, "%c", RDELIM);
	strcat(output, buf);
123
	return path_in(output);
124 125
}

126
/* return the point where two paths intersect.	Assumes that they do. */
127
Point *
128
interpt_pp(p1, p2)
129 130
PATH	   *p1;
PATH	   *p2;
131
{
132

133 134 135 136 137
	Point	   *retval;
	int			i,
				j;
	LSEG		seg1,
				seg2;
138

M
 
Marc G. Fournier 已提交
139
#ifdef NOT_USED
140
	LINE	   *ln;
141

142
#endif
143
	bool		found;			/* We've found the intersection */
144

145
	found = false;				/* Haven't found it yet */
B
Bryan Henderson 已提交
146

147 148 149 150 151 152 153 154
	for (i = 0; i < p1->npts - 1 && !found; i++)
		for (j = 0; j < p2->npts - 1 && !found; j++)
		{
			regress_lseg_construct(&seg1, &p1->p[i], &p1->p[i + 1]);
			regress_lseg_construct(&seg2, &p2->p[j], &p2->p[j + 1]);
			if (lseg_intersect(&seg1, &seg2))
				found = true;
		}
155

M
 
Marc G. Fournier 已提交
156
#ifdef NOT_USED
157 158
	ln = line_construct_pp(&seg2.p[0], &seg2.p[1]);
	retval = interpt_sl(&seg1, ln);
159
#endif
160 161
	retval = lseg_interpt(&seg1, &seg2);

162
	return retval;
163 164 165 166
}


/* like lseg_construct, but assume space already allocated */
167
void
168
regress_lseg_construct(lseg, pt1, pt2)
169 170 171
LSEG	   *lseg;
Point	   *pt1;
Point	   *pt2;
172
{
173 174 175 176 177
	lseg->p[0].x = pt1->x;
	lseg->p[0].y = pt1->y;
	lseg->p[1].x = pt2->x;
	lseg->p[1].y = pt2->y;
	lseg->m = point_sl(pt1, pt2);
178 179
}

180 181
Datum
overpaid(PG_FUNCTION_ARGS)
182
{
183
	TUPLE		tuple = (TUPLE) PG_GETARG_POINTER(0);
184 185
	bool		isnull;
	long		salary;
186

187
	salary = (long) GetAttributeByName(tuple, "salary", &isnull);
188 189 190
	if (isnull)
		PG_RETURN_NULL();
	PG_RETURN_BOOL(salary > 699);
191 192
}

193 194
/* New type "widget"
 * This used to be "circle", but I added circle to builtins,
195
 *	so needed to make sure the names do not collide. - tgl 97/04/21
196 197
 */

198 199
typedef struct
{
200 201 202
	Point		center;
	double		radius;
}			WIDGET;
203

204 205
WIDGET	   *widget_in(char *str);
char	   *widget_out(WIDGET * widget);
206
int			pt_in_widget(Point *point, WIDGET * widget);
207 208 209

#define NARGS	3

210
WIDGET *
211
widget_in(str)
212
char	   *str;
213
{
214 215 216 217 218
	char	   *p,
			   *coord[NARGS],
				buf2[1000];
	int			i;
	WIDGET	   *result;
219 220

	if (str == NULL)
221
		return NULL;
222 223 224 225
	for (i = 0, p = str; *p && i < NARGS && *p != RDELIM; p++)
		if (*p == ',' || (*p == LDELIM && !i))
			coord[i++] = p + 1;
	if (i < NARGS - 1)
226
		return NULL;
227
	result = (WIDGET *) palloc(sizeof(WIDGET));
228 229 230 231
	result->center.x = atof(coord[0]);
	result->center.y = atof(coord[1]);
	result->radius = atof(coord[2]);

232
	sprintf(buf2, "widget_in: read (%f, %f, %f)\n", result->center.x,
233
			result->center.y, result->radius);
234
	return result;
235 236
}

237
char *
238
widget_out(widget)
239
WIDGET	   *widget;
240
{
241
	char	   *result;
242

243
	if (widget == NULL)
244
		return NULL;
245

246 247 248
	result = (char *) palloc(60);
	sprintf(result, "(%g,%g,%g)",
			widget->center.x, widget->center.y, widget->radius);
249
	return result;
250 251
}

252
int
253
pt_in_widget(point, widget)
254 255
Point	   *point;
WIDGET	   *widget;
256
{
257
	extern double point_dt();
258

259
	return point_dt(point, &widget->center) < widget->radius;
260 261 262 263
}

#define ABS(X) ((X) > 0 ? (X) : -(X))

264
int
265 266
boxarea(box)

267
BOX		   *box;
268 269

{
270 271
	int			width,
				height;
272

273
	width = ABS(box->high.x - box->low.x);
274
	height = ABS(box->high.y - box->low.y);
275
	return width * height;
276 277
}

278
char *
279
reverse_name(string)
280
char	   *string;
281
{
282
	int			i;
283 284
	int			len;
	char	   *new_string;
285

286
	if (!(new_string = palloc(NAMEDATALEN)))
287
	{
288
		fprintf(stderr, "reverse_name: palloc failed\n");
289
		return NULL;
290
	}
291 292
	MemSet(new_string, 0, NAMEDATALEN);
	for (i = 0; i < NAMEDATALEN && string[i]; ++i)
293
		;
294
	if (i == NAMEDATALEN || !string[i])
295 296 297 298
		--i;
	len = i;
	for (; i >= 0; --i)
		new_string[len - i] = string[i];
299
	return new_string;
300
}
V
Vadim B. Mikheev 已提交
301 302 303 304 305 306 307 308 309 310

#include "executor/spi.h"		/* this is what you need to work with SPI */
#include "commands/trigger.h"	/* -"- and triggers */

static TransactionId fd17b_xid = InvalidTransactionId;
static TransactionId fd17a_xid = InvalidTransactionId;
static int	fd17b_level = 0;
static int	fd17a_level = 0;
static bool fd17b_recursion = true;
static bool fd17a_recursion = true;
311
extern Datum funny_dup17(PG_FUNCTION_ARGS);
V
Vadim B. Mikheev 已提交
312

313 314
Datum
funny_dup17(PG_FUNCTION_ARGS)
V
Vadim B. Mikheev 已提交
315
{
316
	TriggerData *trigdata = (TriggerData *) fcinfo->context;
V
Vadim B. Mikheev 已提交
317 318 319 320 321 322
	TransactionId *xid;
	int		   *level;
	bool	   *recursion;
	Relation	rel;
	TupleDesc	tupdesc;
	HeapTuple	tuple;
323 324 325
	char	   *query,
			   *fieldval,
			   *fieldtype;
V
Vadim B. Mikheev 已提交
326 327 328 329 330
	char	   *when;
	int			inserted;
	int			selected = 0;
	int			ret;

331 332 333 334 335
	if (!CALLED_AS_TRIGGER(fcinfo))
		elog(ERROR, "funny_dup17: not fired by trigger manager");

	tuple = trigdata->tg_trigtuple;
	rel = trigdata->tg_relation;
V
Vadim B. Mikheev 已提交
336
	tupdesc = rel->rd_att;
337
	if (TRIGGER_FIRED_BEFORE(trigdata->tg_event))
V
Vadim B. Mikheev 已提交
338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361
	{
		xid = &fd17b_xid;
		level = &fd17b_level;
		recursion = &fd17b_recursion;
		when = "BEFORE";
	}
	else
	{
		xid = &fd17a_xid;
		level = &fd17a_level;
		recursion = &fd17a_recursion;
		when = "AFTER ";
	}

	if (!TransactionIdIsCurrentTransactionId(*xid))
	{
		*xid = GetCurrentTransactionId();
		*level = 0;
		*recursion = true;
	}

	if (*level == 17)
	{
		*recursion = false;
362
		return PointerGetDatum(tuple);
V
Vadim B. Mikheev 已提交
363 364 365
	}

	if (!(*recursion))
366
		return PointerGetDatum(tuple);
V
Vadim B. Mikheev 已提交
367 368 369 370 371

	(*level)++;

	SPI_connect();

372 373 374
	fieldval = SPI_getvalue(tuple, tupdesc, 1);
	fieldtype = SPI_gettype(tupdesc, 1);

375
	query = (char *) palloc(100 + NAMEDATALEN * 3 +
376 377 378
							strlen(fieldval) + strlen(fieldtype));

	sprintf(query, "insert into %s select * from %s where %s = '%s'::%s",
V
Vadim B. Mikheev 已提交
379 380
			SPI_getrelname(rel), SPI_getrelname(rel),
			SPI_fname(tupdesc, 1),
381
			fieldval, fieldtype);
V
Vadim B. Mikheev 已提交
382

383
	if ((ret = SPI_exec(query, 0)) < 0)
B
Bruce Momjian 已提交
384
		elog(ERROR, "funny_dup17 (fired %s) on level %3d: SPI_exec (insert ...) returned %d",
V
Vadim B. Mikheev 已提交
385 386 387 388
			 when, *level, ret);

	inserted = SPI_processed;

389
	sprintf(query, "select count (*) from %s where %s = '%s'::%s",
V
Vadim B. Mikheev 已提交
390 391
			SPI_getrelname(rel),
			SPI_fname(tupdesc, 1),
392
			fieldval, fieldtype);
V
Vadim B. Mikheev 已提交
393

394
	if ((ret = SPI_exec(query, 0)) < 0)
B
Bruce Momjian 已提交
395
		elog(ERROR, "funny_dup17 (fired %s) on level %3d: SPI_exec (select ...) returned %d",
V
Vadim B. Mikheev 已提交
396 397 398 399
			 when, *level, ret);

	if (SPI_processed > 0)
	{
400 401
		selected = DatumGetInt32(DirectFunctionCall1(int4in,
								 CStringGetDatum(SPI_getvalue(
B
Bruce Momjian 已提交
402 403 404
									   SPI_tuptable->vals[0],
									   SPI_tuptable->tupdesc,
									   1
405
									   ))));
V
Vadim B. Mikheev 已提交
406 407 408 409 410 411 412 413 414 415 416 417
	}

	elog(NOTICE, "funny_dup17 (fired %s) on level %3d: %d/%d tuples inserted/selected",
		 when, *level, inserted, selected);

	SPI_finish();

	(*level)--;

	if (*level == 0)
		*xid = InvalidTransactionId;

418
	return PointerGetDatum(tuple);
V
Vadim B. Mikheev 已提交
419
}
420

421
extern Datum ttdummy(PG_FUNCTION_ARGS);
422
int32		set_ttdummy(int32 on);
423 424 425

#define TTDUMMY_INFINITY	999999

426 427
static void *splan = NULL;
static bool ttoff = false;
428

429 430
Datum
ttdummy(PG_FUNCTION_ARGS)
431
{
432
	TriggerData *trigdata = (TriggerData *) fcinfo->context;
433 434 435
	Trigger    *trigger;		/* to get trigger name */
	char	  **args;			/* arguments */
	int			attnum[2];		/* fnumbers of start/stop columns */
436 437 438 439
	Datum		oldon,
				oldoff;
	Datum		newon,
				newoff;
440 441 442 443 444 445 446 447 448 449 450 451 452
	Datum	   *cvals;			/* column values */
	char	   *cnulls;			/* column nulls */
	char	   *relname;		/* triggered relation name */
	Relation	rel;			/* triggered relation */
	HeapTuple	trigtuple;
	HeapTuple	newtuple = NULL;
	HeapTuple	rettuple;
	TupleDesc	tupdesc;		/* tuple description */
	int			natts;			/* # of attributes */
	bool		isnull;			/* to know is some column NULL or not */
	int			ret;
	int			i;

453 454 455
	if (!CALLED_AS_TRIGGER(fcinfo))
		elog(ERROR, "ttdummy: not fired by trigger manager");
	if (TRIGGER_FIRED_FOR_STATEMENT(trigdata->tg_event))
B
Bruce Momjian 已提交
456
		elog(ERROR, "ttdummy: can't process STATEMENT events");
457
	if (TRIGGER_FIRED_AFTER(trigdata->tg_event))
B
Bruce Momjian 已提交
458
		elog(ERROR, "ttdummy: must be fired before event");
459
	if (TRIGGER_FIRED_BY_INSERT(trigdata->tg_event))
460
		elog(ERROR, "ttdummy: can't process INSERT event");
461 462
	if (TRIGGER_FIRED_BY_UPDATE(trigdata->tg_event))
		newtuple = trigdata->tg_newtuple;
463

464
	trigtuple = trigdata->tg_trigtuple;
465

466
	rel = trigdata->tg_relation;
467
	relname = SPI_getrelname(rel);
468

469
	/* check if TT is OFF for this relation */
470
	if (ttoff)					/* OFF - nothing to do */
471
	{
472
		pfree(relname);
473
		return PointerGetDatum((newtuple != NULL) ? newtuple : trigtuple);
474
	}
475

476
	trigger = trigdata->tg_trigger;
477 478

	if (trigger->tgnargs != 2)
479 480 481
		elog(ERROR, "ttdummy (%s): invalid (!= 2) number of arguments %d",
			 relname, trigger->tgnargs);

482 483 484
	args = trigger->tgargs;
	tupdesc = rel->rd_att;
	natts = tupdesc->natts;
485 486

	for (i = 0; i < 2; i++)
487
	{
488 489
		attnum[i] = SPI_fnumber(tupdesc, args[i]);
		if (attnum[i] < 0)
B
Bruce Momjian 已提交
490
			elog(ERROR, "ttdummy (%s): there is no attribute %s", relname, args[i]);
491 492 493
		if (SPI_gettypeid(tupdesc, attnum[i]) != INT4OID)
			elog(ERROR, "ttdummy (%s): attributes %s and %s must be of abstime type",
				 relname, args[0], args[1]);
494
	}
495 496

	oldon = SPI_getbinval(trigtuple, tupdesc, attnum[0], &isnull);
497
	if (isnull)
B
Bruce Momjian 已提交
498
		elog(ERROR, "ttdummy (%s): %s must be NOT NULL", relname, args[0]);
499 500

	oldoff = SPI_getbinval(trigtuple, tupdesc, attnum[1], &isnull);
501
	if (isnull)
B
Bruce Momjian 已提交
502
		elog(ERROR, "ttdummy (%s): %s must be NOT NULL", relname, args[1]);
503 504

	if (newtuple != NULL)		/* UPDATE */
505
	{
506
		newon = SPI_getbinval(newtuple, tupdesc, attnum[0], &isnull);
507
		if (isnull)
B
Bruce Momjian 已提交
508
			elog(ERROR, "ttdummy (%s): %s must be NOT NULL", relname, args[0]);
509
		newoff = SPI_getbinval(newtuple, tupdesc, attnum[1], &isnull);
510
		if (isnull)
B
Bruce Momjian 已提交
511
			elog(ERROR, "ttdummy (%s): %s must be NOT NULL", relname, args[1]);
512 513 514 515 516 517

		if (oldon != newon || oldoff != newoff)
			elog(ERROR, "ttdummy (%s): you can't change %s and/or %s columns (use set_ttdummy)",
				 relname, args[0], args[1]);

		if (newoff != TTDUMMY_INFINITY)
518
		{
519
			pfree(relname);		/* allocated in upper executor context */
520
			return PointerGetDatum(NULL);
521 522 523 524
		}
	}
	else if (oldoff != TTDUMMY_INFINITY)		/* DELETE */
	{
525
		pfree(relname);
526
		return PointerGetDatum(NULL);
527
	}
528

529
	{
530
		text   *seqname = textin("ttdummy_seq");
531

532 533
		newoff = DirectFunctionCall1(nextval,
									 PointerGetDatum(seqname));
534
		pfree(seqname);
535
	}
536

537 538
	/* Connect to SPI manager */
	if ((ret = SPI_connect()) < 0)
B
Bruce Momjian 已提交
539
		elog(ERROR, "ttdummy (%s): SPI_connect returned %d", relname, ret);
540

541
	/* Fetch tuple values and nulls */
542 543
	cvals = (Datum *) palloc(natts * sizeof(Datum));
	cnulls = (char *) palloc(natts * sizeof(char));
544 545
	for (i = 0; i < natts; i++)
	{
546 547
		cvals[i] = SPI_getbinval((newtuple != NULL) ? newtuple : trigtuple,
								 tupdesc, i + 1, &isnull);
548 549
		cnulls[i] = (isnull) ? 'n' : ' ';
	}
550

551
	/* change date column(s) */
552
	if (newtuple)				/* UPDATE */
553
	{
554
		cvals[attnum[0] - 1] = newoff;	/* start_date eq current date */
555
		cnulls[attnum[0] - 1] = ' ';
556
		cvals[attnum[1] - 1] = TTDUMMY_INFINITY;		/* stop_date eq INFINITY */
557 558
		cnulls[attnum[1] - 1] = ' ';
	}
559 560
	else
/* DELETE */
561
	{
562
		cvals[attnum[1] - 1] = newoff;	/* stop_date eq current date */
563 564
		cnulls[attnum[1] - 1] = ' ';
	}
565

566 567 568 569 570
	/* if there is no plan ... */
	if (splan == NULL)
	{
		void	   *pplan;
		Oid		   *ctypes;
571
		char	   *query;
572

573
		/* allocate space in preparation */
574
		ctypes = (Oid *) palloc(natts * sizeof(Oid));
575
		query = (char *) palloc(100 + 16 * natts);
576

577
		/*
578
		 * Construct query: INSERT INTO _relation_ VALUES ($1, ...)
579
		 */
580
		sprintf(query, "INSERT INTO %s VALUES (", relname);
581 582
		for (i = 1; i <= natts; i++)
		{
583
			sprintf(query + strlen(query), "$%d%s",
584
					i, (i < natts) ? ", " : ")");
585 586
			ctypes[i - 1] = SPI_gettypeid(tupdesc, i);
		}
587

588
		/* Prepare plan for query */
589
		pplan = SPI_prepare(query, natts, ctypes);
590
		if (pplan == NULL)
B
Bruce Momjian 已提交
591
			elog(ERROR, "ttdummy (%s): SPI_prepare returned %d", relname, SPI_result);
592

593 594
		pplan = SPI_saveplan(pplan);
		if (pplan == NULL)
B
Bruce Momjian 已提交
595
			elog(ERROR, "ttdummy (%s): SPI_saveplan returned %d", relname, SPI_result);
596

597 598
		splan = pplan;
	}
599

600
	ret = SPI_execp(splan, cvals, cnulls, 0);
601

602
	if (ret < 0)
B
Bruce Momjian 已提交
603
		elog(ERROR, "ttdummy (%s): SPI_execp returned %d", relname, ret);
604

605
	/* Tuple to return to upper Executor ... */
606
	if (newtuple)				/* UPDATE */
607 608
	{
		HeapTuple	tmptuple;
609 610 611

		tmptuple = SPI_copytuple(trigtuple);
		rettuple = SPI_modifytuple(rel, tmptuple, 1, &(attnum[1]), &newoff, NULL);
612
		SPI_freetuple(tmptuple);
613
	}
614 615
	else
/* DELETE */
616
		rettuple = trigtuple;
617 618 619 620

	SPI_finish();				/* don't forget say Bye to SPI mgr */

	pfree(relname);
621

622
	return PointerGetDatum(rettuple);
623 624 625 626 627
}

int32
set_ttdummy(int32 on)
{
628 629

	if (ttoff)					/* OFF currently */
630 631
	{
		if (on == 0)
632
			return 0;
633

634 635
		/* turn ON */
		ttoff = false;
636
		return 0;
637
	}
638

639 640
	/* ON currently */
	if (on != 0)
641
		return 1;
642

643 644
	/* turn OFF */
	ttoff = true;
645

646
	return 1;
647 648

}