420 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
			
		
		
	
	
			420 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
/*
 | 
						||
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
 | 
						||
 *
 | 
						||
 * This program is free software: you can use, redistribute, and/or modify
 | 
						||
 * it under the terms of the GNU Affero General Public License, version 3
 | 
						||
 * or later ("AGPL"), as published by the Free Software Foundation.
 | 
						||
 *
 | 
						||
 * This program is distributed in the hope that it will be useful, but WITHOUT
 | 
						||
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | 
						||
 * FITNESS FOR A PARTICULAR PURPOSE.
 | 
						||
 *
 | 
						||
 * You should have received a copy of the GNU Affero General Public License
 | 
						||
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 | 
						||
 */
 | 
						||
 | 
						||
using System;
 | 
						||
using System.Text;
 | 
						||
using System.Collections.Generic;
 | 
						||
using System.Runtime.InteropServices;
 | 
						||
using System.Collections;
 | 
						||
 | 
						||
namespace TDengineDriver
 | 
						||
{
 | 
						||
    class TDengineTest
 | 
						||
    {
 | 
						||
        //connect parameters
 | 
						||
        private string host;
 | 
						||
        private string configDir;
 | 
						||
        private string user;
 | 
						||
        private string password;
 | 
						||
        private short port = 0;
 | 
						||
 | 
						||
        //sql parameters
 | 
						||
        private string dbName;
 | 
						||
        private string tbName;
 | 
						||
        private string precision;
 | 
						||
 | 
						||
        private bool isInsertData;
 | 
						||
        private bool isQueryData;
 | 
						||
 | 
						||
        private long tableCount;
 | 
						||
        private long totalRows;
 | 
						||
        private long batchRows;
 | 
						||
        private long beginTimestamp = 1551369600000L;
 | 
						||
 | 
						||
        private IntPtr conn = IntPtr.Zero;
 | 
						||
        private long rowsInserted = 0;
 | 
						||
 | 
						||
        static void Main(string[] args)
 | 
						||
        {
 | 
						||
            TDengineTest tester = new TDengineTest();
 | 
						||
            tester.ReadArgument(args);
 | 
						||
 | 
						||
 | 
						||
            tester.InitTDengine();
 | 
						||
            tester.ConnectTDengine();
 | 
						||
            tester.createDatabase();
 | 
						||
            tester.useDatabase();
 | 
						||
            tester.checkDropTable();
 | 
						||
            tester.createTable();
 | 
						||
            tester.checkInsert();
 | 
						||
            tester.checkSelect();
 | 
						||
            tester.checkDropTable();
 | 
						||
            tester.dropDatabase();
 | 
						||
            tester.CloseConnection();
 | 
						||
            tester.cleanup();
 | 
						||
 | 
						||
 | 
						||
        }
 | 
						||
 | 
						||
        public long GetArgumentAsLong(String[] argv, String argName, int minVal, int maxVal, int defaultValue)
 | 
						||
        {
 | 
						||
            int argc = argv.Length;
 | 
						||
            for (int i = 0; i < argc; ++i)
 | 
						||
            {
 | 
						||
                if (argName != argv[i])
 | 
						||
                {
 | 
						||
                    continue;
 | 
						||
                }
 | 
						||
                if (i < argc - 1)
 | 
						||
                {
 | 
						||
                    String tmp = argv[i + 1];
 | 
						||
                    if (tmp[0] == '-')
 | 
						||
                    {
 | 
						||
                        Console.WriteLine("option {0:G} requires an argument", tmp);
 | 
						||
                        ExitProgram();
 | 
						||
                    }
 | 
						||
 | 
						||
                    long tmpVal = Convert.ToInt64(tmp);
 | 
						||
                    if (tmpVal < minVal || tmpVal > maxVal)
 | 
						||
                    {
 | 
						||
                        Console.WriteLine("option {0:G} should in range [{1:G}, {2:G}]", argName, minVal, maxVal);
 | 
						||
                        ExitProgram();
 | 
						||
                    }
 | 
						||
 | 
						||
                    return tmpVal;
 | 
						||
                }
 | 
						||
            }
 | 
						||
 | 
						||
            return defaultValue;
 | 
						||
        }
 | 
						||
 | 
						||
        public String GetArgumentAsString(String[] argv, String argName, String defaultValue)
 | 
						||
        {
 | 
						||
            int argc = argv.Length;
 | 
						||
            for (int i = 0; i < argc; ++i)
 | 
						||
            {
 | 
						||
                if (argName != argv[i])
 | 
						||
                {
 | 
						||
                    continue;
 | 
						||
                }
 | 
						||
                if (i < argc - 1)
 | 
						||
                {
 | 
						||
                    String tmp = argv[i + 1];
 | 
						||
                    if (tmp[0] == '-')
 | 
						||
                    {
 | 
						||
                        Console.WriteLine("option {0:G} requires an argument", tmp);
 | 
						||
                        ExitProgram();
 | 
						||
                    }
 | 
						||
                    return tmp;
 | 
						||
                }
 | 
						||
            }
 | 
						||
 | 
						||
            return defaultValue;
 | 
						||
        }
 | 
						||
 | 
						||
        public void PrintHelp(String[] argv)
 | 
						||
        {
 | 
						||
            for (int i = 0; i < argv.Length; ++i)
 | 
						||
            {
 | 
						||
                if ("--help" == argv[i])
 | 
						||
                {
 | 
						||
                    String indent = "    ";
 | 
						||
                    Console.WriteLine("taosTest is simple example to operate TDengine use C# Language.\n");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-h");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "TDEngine server IP address to connect");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-u");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "The TDEngine user name to use when connecting to the server, default is root");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-p");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "The TDEngine user name to use when connecting to the server, default is taosdata");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-d");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Database used to create table or import data, default is db");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-s");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Super Tables used to create table, default is mt");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-t");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Table prefixs, default is t");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-w");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Whether to insert data");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-r");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Whether to query data");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-n");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "How many Tables to create, default is 10");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-b");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "How many rows per insert batch, default is 10");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-i");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "How many rows to insert, default is 100");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-c");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Configuration directory");
 | 
						||
                    //
 | 
						||
                    Console.WriteLine("{0:G}{1:G}", indent, "-ps");
 | 
						||
                    Console.WriteLine("{0:G}{1:G}{2:G}", indent, indent, "Configurate db precision,default millisecond");
 | 
						||
                    ExitProgram();
 | 
						||
                }
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
        public void ReadArgument(String[] argv)
 | 
						||
        {
 | 
						||
            PrintHelp(argv);
 | 
						||
            host = this.GetArgumentAsString(argv, "-h", "127.0.0.1");
 | 
						||
            user = this.GetArgumentAsString(argv, "-u", "root");
 | 
						||
            password = this.GetArgumentAsString(argv, "-p", "taosdata");
 | 
						||
            dbName = this.GetArgumentAsString(argv, "-d", "test");
 | 
						||
            tbName = this.GetArgumentAsString(argv, "-s", "weather");
 | 
						||
            precision = this.GetArgumentAsString(argv, "-ps", "ms");
 | 
						||
            isInsertData = this.GetArgumentAsLong(argv, "-w", 0, 1, 1) != 0;
 | 
						||
            isQueryData = this.GetArgumentAsLong(argv, "-r", 0, 1, 1) != 0;
 | 
						||
            tableCount = this.GetArgumentAsLong(argv, "-n", 1, 10000, 10);
 | 
						||
            batchRows = this.GetArgumentAsLong(argv, "-b", 1, 1000, 500);
 | 
						||
            totalRows = this.GetArgumentAsLong(argv, "-i", 1, 10000000, 10000);
 | 
						||
            configDir = this.GetArgumentAsString(argv, "-c", "C:/TDengine/cfg");
 | 
						||
        }
 | 
						||
 | 
						||
        public void InitTDengine()
 | 
						||
        {
 | 
						||
            TDengine.Options((int)TDengineInitOption.TDDB_OPTION_CONFIGDIR, this.configDir);
 | 
						||
            TDengine.Options((int)TDengineInitOption.TDDB_OPTION_SHELL_ACTIVITY_TIMER, "60");
 | 
						||
            Console.WriteLine("init...");
 | 
						||
            TDengine.Init();
 | 
						||
            Console.WriteLine("get connection starting...");
 | 
						||
        }
 | 
						||
 | 
						||
        public void ConnectTDengine()
 | 
						||
        {
 | 
						||
            string db = "";
 | 
						||
            this.conn = TDengine.Connect(this.host, this.user, this.password, db, this.port);
 | 
						||
            if (this.conn == IntPtr.Zero)
 | 
						||
            {
 | 
						||
                Console.WriteLine("connection failed: " + this.host);
 | 
						||
                ExitProgram();
 | 
						||
            }
 | 
						||
            else
 | 
						||
            {
 | 
						||
                Console.WriteLine("[ OK ] Connection established.");
 | 
						||
            }
 | 
						||
        }
 | 
						||
        public void createDatabase()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("create database if not exists ").Append(this.dbName).Append(" precision '").Append(this.precision).Append("'");
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void useDatabase()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("use ").Append(this.dbName);
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void checkSelect()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("select * from ").Append(this.dbName).Append(".").Append(this.tbName);
 | 
						||
            ExecuteQuery(sql.ToString());
 | 
						||
        }
 | 
						||
        public void createTable()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("create table if not exists ").Append(this.dbName).Append(".").Append(this.tbName).Append("(ts timestamp, temperature float, humidity int)");
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void checkInsert()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("insert into ").Append(this.dbName).Append(".").Append(this.tbName).Append("(ts, temperature, humidity) values(now, 20.5, 34)");
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void checkDropTable()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("drop table if exists ").Append(this.dbName).Append(".").Append(this.tbName).Append("");
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void dropDatabase()
 | 
						||
        {
 | 
						||
            StringBuilder sql = new StringBuilder();
 | 
						||
            sql.Append("drop database if exists ").Append(this.dbName);
 | 
						||
            execute(sql.ToString());
 | 
						||
        }
 | 
						||
        public void execute(string sql)
 | 
						||
        {
 | 
						||
            DateTime dt1 = DateTime.Now;
 | 
						||
            IntPtr res = TDengine.Query(this.conn, sql.ToString());
 | 
						||
            DateTime dt2 = DateTime.Now;
 | 
						||
            TimeSpan span = dt2 - dt1;
 | 
						||
 | 
						||
            if ((res == IntPtr.Zero) || (TDengine.ErrorNo(res) != 0))
 | 
						||
            {
 | 
						||
              Console.Write(sql.ToString() + " failure, ");
 | 
						||
              if (res != IntPtr.Zero) {
 | 
						||
                Console.Write("reason: " + TDengine.Error(res));
 | 
						||
              }
 | 
						||
              Console.WriteLine("");
 | 
						||
              ExitProgram();
 | 
						||
            }
 | 
						||
            else
 | 
						||
            {
 | 
						||
                Console.WriteLine(sql.ToString() + " success");
 | 
						||
            }
 | 
						||
            TDengine.FreeResult(res);
 | 
						||
        }
 | 
						||
 | 
						||
        public void ExecuteQuery(string sql)
 | 
						||
        {
 | 
						||
 | 
						||
            DateTime dt1 = DateTime.Now;
 | 
						||
            long queryRows = 0;
 | 
						||
            IntPtr res = TDengine.Query(conn, sql);
 | 
						||
            getPrecision(res);
 | 
						||
            if ((res == IntPtr.Zero) || (TDengine.ErrorNo(res) != 0))
 | 
						||
            {
 | 
						||
              Console.Write(sql.ToString() + " failure, ");
 | 
						||
              if (res != IntPtr.Zero) {
 | 
						||
                Console.Write("reason: " + TDengine.Error(res));
 | 
						||
              }
 | 
						||
              Console.WriteLine("");
 | 
						||
              ExitProgram();
 | 
						||
            }
 | 
						||
            DateTime dt2 = DateTime.Now;
 | 
						||
            TimeSpan span = dt2 - dt1;
 | 
						||
            Console.WriteLine("[OK] time cost: " + span.ToString() + "ms, execute statement ====> " + sql.ToString());
 | 
						||
            int fieldCount = TDengine.FieldCount(res);
 | 
						||
 | 
						||
            List<TDengineMeta> metas = TDengine.FetchFields(res);
 | 
						||
            for (int j = 0; j < metas.Count; j++)
 | 
						||
            {
 | 
						||
                TDengineMeta meta = (TDengineMeta)metas[j];
 | 
						||
            }
 | 
						||
 | 
						||
            IntPtr rowdata;
 | 
						||
            StringBuilder builder = new StringBuilder();
 | 
						||
            while ((rowdata = TDengine.FetchRows(res)) != IntPtr.Zero)
 | 
						||
            {
 | 
						||
                queryRows++;
 | 
						||
                for (int fields = 0; fields < fieldCount; ++fields)
 | 
						||
                {
 | 
						||
                    TDengineMeta meta = metas[fields];
 | 
						||
                    int offset = IntPtr.Size * fields;
 | 
						||
                    IntPtr data = Marshal.ReadIntPtr(rowdata, offset);
 | 
						||
 | 
						||
                    builder.Append("---");
 | 
						||
 | 
						||
                    if (data == IntPtr.Zero)
 | 
						||
                    {
 | 
						||
                        builder.Append("NULL");
 | 
						||
                        continue;
 | 
						||
                    }
 | 
						||
 | 
						||
                    switch ((TDengineDataType)meta.type)
 | 
						||
                    {
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_BOOL:
 | 
						||
                            bool v1 = Marshal.ReadByte(data) == 0 ? false : true;
 | 
						||
                            builder.Append(v1);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_TINYINT:
 | 
						||
                            byte v2 = Marshal.ReadByte(data);
 | 
						||
                            builder.Append(v2);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_SMALLINT:
 | 
						||
                            short v3 = Marshal.ReadInt16(data);
 | 
						||
                            builder.Append(v3);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_INT:
 | 
						||
                            int v4 = Marshal.ReadInt32(data);
 | 
						||
                            builder.Append(v4);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_BIGINT:
 | 
						||
                            long v5 = Marshal.ReadInt64(data);
 | 
						||
                            builder.Append(v5);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_FLOAT:
 | 
						||
                            float v6 = (float)Marshal.PtrToStructure(data, typeof(float));
 | 
						||
                            builder.Append(v6);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_DOUBLE:
 | 
						||
                            double v7 = (double)Marshal.PtrToStructure(data, typeof(double));
 | 
						||
                            builder.Append(v7);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_BINARY:
 | 
						||
                            string v8 = Marshal.PtrToStringAnsi(data);
 | 
						||
                            builder.Append(v8);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_TIMESTAMP:
 | 
						||
                            long v9 = Marshal.ReadInt64(data);
 | 
						||
                            builder.Append(v9);
 | 
						||
                            break;
 | 
						||
                        case TDengineDataType.TSDB_DATA_TYPE_NCHAR:
 | 
						||
                            string v10 = Marshal.PtrToStringAnsi(data);
 | 
						||
                            builder.Append(v10);
 | 
						||
                            break;
 | 
						||
                    }
 | 
						||
                }
 | 
						||
                builder.Append("---");
 | 
						||
 | 
						||
                if (queryRows <= 10)
 | 
						||
                {
 | 
						||
                    Console.WriteLine(builder.ToString());
 | 
						||
                }
 | 
						||
                builder.Clear();
 | 
						||
            }
 | 
						||
 | 
						||
            if (TDengine.ErrorNo(res) != 0)
 | 
						||
            {
 | 
						||
                Console.Write("Query is not complete, Error {0:G}", TDengine.ErrorNo(res), TDengine.Error(res));
 | 
						||
            }
 | 
						||
            Console.WriteLine("");
 | 
						||
 | 
						||
            TDengine.FreeResult(res);
 | 
						||
 | 
						||
        }
 | 
						||
 | 
						||
        public void CloseConnection()
 | 
						||
        {
 | 
						||
            if (this.conn != IntPtr.Zero)
 | 
						||
            {
 | 
						||
                TDengine.Close(this.conn);
 | 
						||
                Console.WriteLine("connection closed.");
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
        static void ExitProgram()
 | 
						||
        {
 | 
						||
            System.Environment.Exit(0);
 | 
						||
        }
 | 
						||
 | 
						||
        public void cleanup()
 | 
						||
        {
 | 
						||
            Console.WriteLine("clean up...");
 | 
						||
            System.Environment.Exit(0);
 | 
						||
        }
 | 
						||
        // method to get db precision
 | 
						||
        public void getPrecision(IntPtr res)
 | 
						||
        {
 | 
						||
            int psc=TDengine.ResultPrecision(res);
 | 
						||
            switch(psc)
 | 
						||
            {
 | 
						||
                case 0:
 | 
						||
                    Console.WriteLine("db:[{0:G}]'s precision is {1:G}",this.dbName,"millisecond");
 | 
						||
                    break;
 | 
						||
                case 1:
 | 
						||
                    Console.WriteLine("db:[{0:G}]'s precision is {1:G}",this.dbName,"microsecond");
 | 
						||
                    break;
 | 
						||
                case 2:
 | 
						||
                    Console.WriteLine("db:[{0:G}]'s precision is {1:G}",this.dbName,"nanosecond");
 | 
						||
                    break;
 | 
						||
            }
 | 
						||
 | 
						||
        }
 | 
						||
    }
 | 
						||
}
 |