1 /** 2 * ae.demo.sqlite.exec 3 * 4 * License: 5 * This Source Code Form is subject to the terms of 6 * the Mozilla Public License, v. 2.0. If a copy of 7 * the MPL was not distributed with this file, You 8 * can obtain one at http://mozilla.org/MPL/2.0/. 9 * 10 * Authors: 11 * Vladimir Panteleev <vladimir@thecybershadow.net> 12 */ 13 14 module ae.demo.sqlite.exec; 15 16 import std.stdio; 17 import std.algorithm; 18 import std.array; 19 import std.conv; 20 import std.datetime; 21 22 import ae.sys.sqlite3; 23 import ae.sys.console; 24 25 void main(string[] args) 26 { 27 if (args.length < 2 || args.length > 3) 28 return stderr.writeln("Usage: exec DATABASE [COMMAND]"); 29 30 auto db = new SQLite(args[1]); 31 32 void exec(string sql, bool interactive) 33 { 34 int idx = 0; 35 string[][] rows; 36 StopWatch sw; 37 sw.start(); 38 foreach (cells, columns; db.query(sql)) 39 { 40 sw.stop(); 41 if (rows is null) 42 rows ~= ["#"] ~ array(map!`a.idup`(columns)); 43 rows ~= [to!string(idx++)] ~ array(map!`a.idup`(cells)); 44 sw.start(); 45 } 46 sw.stop(); 47 stderr.writeln("Query executed in ", dur!"hnsecs"(sw.peek().hnsecs).toString().replace("μ", "u")); 48 49 if (rows.length == 0) 50 return; 51 auto widths = new size_t[rows[0].length]; 52 foreach (row; rows) 53 { 54 assert(row.length == rows[0].length); 55 foreach (i, cell; row) 56 widths[i] = max(widths[i], textWidth(cell)); 57 } 58 foreach (j, row; rows) 59 { 60 auto rowLines = array(map!textLines(row)); 61 auto lineCount = reduce!max(map!`a.length`(rowLines)); 62 63 foreach (line; 0..lineCount) 64 { 65 foreach (i, lines; rowLines) 66 { 67 if (i) write(" │ "); 68 string col = line < lines.length ? lines[line] : null; 69 write(col, std.array.replicate(" ", widths[i]-col.length)); 70 } 71 writeln(); 72 } 73 if (j==0) 74 { 75 foreach (i, w; widths) 76 { 77 if (i) write("─┼─"); 78 write(std.array.replicate("─", w)); 79 } 80 writeln(); 81 } 82 } 83 writeln(); 84 } 85 86 if (args.length == 3) 87 exec(args[2], false); 88 else 89 while (!stdin.eof()) 90 { 91 write("sqlite> "); 92 stdout.flush(); 93 try 94 exec(readln().chomp(), true); 95 catch (Exception e) 96 writeln(e.msg); 97 } 98 } 99 100 import std.utf; 101 import std..string; 102 103 size_t textWidth(string s) 104 { 105 try 106 { 107 auto lines = splitLines(s); 108 size_t w = 0; 109 foreach (line; lines) 110 w = max(w, std.utf.count(line)); 111 return w; 112 } 113 catch (Exception e) 114 return s.length; 115 } 116 117 string[] textLines(string s) 118 { 119 try 120 return splitLines(s); 121 catch (Exception e) 122 return [s]; 123 }