[GRASS-SVN] r32482 - grass/trunk/doc/vector/v.example
svn_grass at osgeo.org
svn_grass at osgeo.org
Sun Aug 3 06:12:50 EDT 2008
Author: neteler
Date: 2008-08-03 06:12:49 -0400 (Sun, 03 Aug 2008)
New Revision: 32482
Modified:
grass/trunk/doc/vector/v.example/main.c
Log:
Maris Nartiss: improved version with DB support
Modified: grass/trunk/doc/vector/v.example/main.c
===================================================================
--- grass/trunk/doc/vector/v.example/main.c 2008-08-03 10:06:45 UTC (rev 32481)
+++ grass/trunk/doc/vector/v.example/main.c 2008-08-03 10:12:49 UTC (rev 32482)
@@ -2,27 +2,25 @@
*
* MODULE: v.example
*
- * AUTHOR(S): GRASS Development Team
+ * AUTHOR(S): GRASS Development Team, Radim Blazek, Maris Nartiss
*
- * PURPOSE: example vector module does something like:
- * v.llabel -i map=m1 value=1
- * but the new map is written instead of update of
- * the old one
+ * PURPOSE: copies vector data from source map to destination map
+ * prints out all point coordinates and atributes
*
- * COPYRIGHT: (C) 2002-2005 by the GRASS Development Team
+ * COPYRIGHT: (C) 2002-2008 by the GRASS Development Team
*
* This program is free software under the
* GNU General Public License (>=v2).
* Read the file COPYING that comes with GRASS
* for details.
*
- * TODO: - add DB support
****************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <grass/gis.h>
#include <grass/Vect.h>
+#include <grass/dbmi.h>
#include <grass/glocale.h>
int main(int argc, char *argv[])
@@ -30,13 +28,21 @@
struct Map_info In, Out;
static struct line_pnts *Points;
struct line_cats *Cats;
- int i, type, cat;
- char *mapset;
+ int i, type, cat, ncols, nrows, col, more, open3d;
+ char *mapset, sql[200];
struct GModule *module; /* GRASS module for parsing arguments */
struct Option *old, *new;
+ dbDriver *driver;
+ dbHandle handle;
+ dbCursor cursor;
+ dbTable *table;
+ dbColumn *column;
+ dbString table_name, dbsql, valstr;
+ struct field_info *Fi, *Fin;
/* initialize GIS environment */
- G_gisinit(argv[0]); /* reads grass env, stores program name to G_program_name() */
+ /* reads grass env, stores program name to G_program_name() */
+ G_gisinit(argv[0]);
/* initialize module */
module = G_define_module();
@@ -50,44 +56,150 @@
/* options and flags parser */
if (G_parser(argc, argv))
- exit(EXIT_FAILURE);
+ exit(EXIT_FAILURE);
+ /* Create and initialize struct's where to store points/lines and categories */
Points = Vect_new_line_struct();
Cats = Vect_new_cats_struct();
- Vect_check_input_output_name(new->answer, old->answer, GV_FATAL_EXIT);
+ /* Check 1) output is legal vector name; 2) if can find input map;
+ 3) if input was found in current mapset, check if input != output.
+ lib/vector/Vlib/legal_vname.c
+ */
+ Vect_check_input_output_name(old->answer, new->answer, GV_FATAL_EXIT);
if ((mapset = G_find_vector2(old->answer, "")) == NULL)
- G_fatal_error(_("Vector map <%s> not found"), old->answer);
+ G_fatal_error(_("Vector map <%s> not found"), old->answer);
- Vect_set_open_level(2);
+ /* Predetermine level at which a map will be opened for reading
+ lib/vector/Vlib/open.c
+ */
+ if (Vect_set_open_level(2))
+ G_fatal_error(_("Unable to set predetermined vector open level"));
+ /* Open existing vector for reading
+ lib/vector/Vlib/open.c
+ */
if (1 > Vect_open_old(&In, old->answer, mapset))
- G_fatal_error(_("Unable to open vector map <%s>"), old->answer);
+ G_fatal_error(_("Unable to open vector map <%s>"), old->answer);
+
+ /* Check if old vector is 3D. We should preserve 3D data. */
+ if (Vect_is_3d(&In)) open3d = WITH_Z;
+ else open3d = WITHOUT_Z;
+
+ /* Open new vector for reading/writing */
+ if (0 > Vect_open_new(&Out, new->answer, open3d)) {
+ Vect_close(&In);
+ G_fatal_error(_("Unable to create vector map <%s>"), new->answer);
+ }
- if (0 > Vect_open_new(&Out, new->answer, WITHOUT_Z)) {
- Vect_close(&In);
- G_fatal_error(_("Unable to create vector map <%s>"), new->answer);
- }
+ /* Let's get vector layers db connections information */
+ Fi = Vect_get_field(&In, 1);
+ if (!Fi) {
+ Vect_close(&In);
+ G_fatal_error(_("Database connection not defined for layer %d"), 1);
+ }
+ /* Output information usefull for debuging
+ incluse/vect/dig_structs.h
+ */
+ G_debug(1,"Field number:%d; Name:<%s>; Driver:<%s>; Database:<%s>; Table:<%s>; Key:<%s>;\n",
+ Fi->number, Fi->name, Fi->driver, Fi->database, Fi->table, Fi->key);
+
+ /* Prepeare strings for use in db_* calls */
+ db_init_string(&dbsql);
+ db_init_string(&valstr);
+ db_init_string(&table_name);
+ db_init_handle(&handle);
+
+ /* Prepearing database for use */
+ driver = db_start_driver(Fi->driver);
+ if (driver == NULL) {
+ Vect_close(&In);
+ G_fatal_error(_("Unable to start driver <%s>"), Fi->driver);
+ }
+ db_set_handle(&handle, Fi->database, NULL);
+ if (db_open_database(driver, &handle) != DB_OK) {
+ Vect_close(&In);
+ G_fatal_error(_("Unable to open database <%s> by driver <%s>"), Fi->database, driver);
+ }
+ db_set_string(&table_name, Fi->table);
+ if (db_describe_table (driver, &table_name, &table) != DB_OK) {
+ Vect_close(&In);
+ G_fatal_error (_("Unable to describe table <%s>"), Fi->table);
+ }
+ ncols = db_get_table_number_of_columns(table);
- Vect_copy_head_data(&In, &Out);
- Vect_hist_copy(&In, &Out);
- Vect_hist_command(&Out);
+ /* Copy header and history data from old to new map */
+ Vect_copy_head_data(&In, &Out);
+ Vect_hist_copy(&In, &Out);
+ Vect_hist_command(&Out);
- i = 1;
- while ((type = Vect_read_next_line(&In, Points, Cats)) > 0) {
- if (type == GV_LINE) {
- if (Vect_cat_get(Cats, 1, &cat) == 0) {
- Vect_cat_set(Cats, 1, i);
- i++;
- }
+ i = 1;
+ /* Let's do something with every vector feature in input map... */
+ /* Read in single line and get it's type */
+ while ((type = Vect_read_next_line(&In, Points, Cats)) > 0) {
+ /* If Points contain line... */
+ if (type == GV_LINE || type == GV_POINT || type == GV_CENTROID) {
+ if (Vect_cat_get(Cats, 1, &cat) == 0) {
+ Vect_cat_set(Cats, 1, i);
+ i++;
+ }
+ }
+ if (type == GV_POINT) {
+ /* Print out point coordinates */
+ printf("No:%d\tX:%f\tY:%f\tZ:%f\tCAT:%d\n",i,*Points->x,*Points->y,*Points->z,cat);
+
+ /* Prepeare SQL query to get point atribute data */
+ sprintf(sql,"select * from %s where %s=%d",Fi->table,Fi->key,cat);
+ G_debug(1,"SQL: \"%s\"",sql);
+ db_set_string(&dbsql, sql);
+ /* Now execute query */
+ if (db_open_select_cursor(driver, &dbsql, &cursor, DB_SEQUENTIAL) != DB_OK)
+ G_warning(_("Unabale to get attribute data for cat %d"), cat);
+ else {
+ /* Result count */
+ nrows = db_get_num_rows(&cursor);
+ G_debug(1,"Result count: %d",nrows);
+ table = db_get_cursor_table (&cursor);
+ /* Let's output every columns name and value */
+ while(1) {
+ if (db_fetch (&cursor, DB_NEXT, &more) != DB_OK) {
+ G_warning(_("Error while retreiving database record for cat %d"), cat);
+ break;
+ }
+ if (!more) break;
+ for (col = 0; col < ncols; col++) {
+ column = db_get_table_column(table, col);
+ db_convert_column_value_to_string(column, &valstr);
+ printf("%s: %s\t", db_get_column_name(column), db_get_string(&valstr));
+ }
+ printf("\n");
+ }
+ db_close_cursor(&cursor);
+ }
+ }
+ /* Only now we write data into new vector */
+ Vect_write_line(&Out, type, Points, Cats);
}
- Vect_write_line(&Out, type, Points, Cats);
- }
-
+
+ /* Create database for new vector map */
+ Fin = Vect_default_field_info(&Out, 1, NULL, GV_1TABLE);
+ driver = db_start_driver_open_database(Fin->driver, Fin->database);
+ G_debug(1,"Field number:%d; Name:<%s>; Driver:<%s>; Database:<%s>; Table:<%s>; Key:<%s>;\n",
+ Fin->number, Fin->name, Fin->driver, Fin->database, Fin->table, Fin->key);
+
+ /* Let's copy atribute table data */
+ if (db_copy_table(Fi->driver,Fi->database,Fi->table,
+ Fin->driver,Vect_subst_var(Fin->database,&Out),Fin->table) == DB_FAILED)
+ G_warning(_("Unable to copy atribute table to vector map <%s>"), new->answer);
+ else Vect_map_add_dblink(&Out, Fin->number, Fin->name, Fin->table, Fi->key, Fin->database, Fin->driver);
+
+ /* Build topology for vector map and close them */
Vect_build(&Out, stdout);
Vect_close(&In);
Vect_close(&Out);
+ db_close_database_shutdown_driver(driver);
+ /* Don't forget to report to caller sucessfull end of data processing :) */
exit(EXIT_SUCCESS);
}
More information about the grass-commit
mailing list