mirror of https://github.com/adamdruppe/arsd.git
mysql.d improvements
This commit is contained in:
parent
62dbfa5bdc
commit
4b728fbb2f
88
cgi.d
88
cgi.d
|
@ -129,37 +129,95 @@
|
||||||
Guide_for_PHP_users:
|
Guide_for_PHP_users:
|
||||||
If you are coming from PHP, here's a quick guide to help you get started:
|
If you are coming from PHP, here's a quick guide to help you get started:
|
||||||
|
|
||||||
```
|
$(SIDE_BY_SIDE
|
||||||
$_GET["var"] == cgi.get["var"]
|
$(COLUMN
|
||||||
$_POST["var"] == cgi.post["var"]
|
```php
|
||||||
$_COOKIE["var"] == cgi.cookies["var"]
|
<?php
|
||||||
```
|
$foo = $_POST["foo"];
|
||||||
|
$bar = $_GET["bar"];
|
||||||
|
$baz = $_COOKIE["baz"];
|
||||||
|
|
||||||
|
$user_ip = $_SERVER["REMOTE_ADDR"];
|
||||||
|
$host = $_SERVER["HTTP_HOST"];
|
||||||
|
$path = $_SERVER["PATH_INFO"];
|
||||||
|
|
||||||
|
setcookie("baz", "some value");
|
||||||
|
|
||||||
|
echo "hello!";
|
||||||
|
?>
|
||||||
|
```
|
||||||
|
)
|
||||||
|
$(COLUMN
|
||||||
|
---
|
||||||
|
import arsd.cgi;
|
||||||
|
void app(Cgi cgi) {
|
||||||
|
string foo = cgi.post["foo"];
|
||||||
|
string bar = cgi.get["bar"];
|
||||||
|
string baz = cgi.cookies["baz"];
|
||||||
|
|
||||||
|
string user_ip = cgi.remoteAddress;
|
||||||
|
string host = cgi.host;
|
||||||
|
string path = cgi.pathInfo;
|
||||||
|
|
||||||
|
cgi.setCookie("baz", "some value");
|
||||||
|
|
||||||
|
cgi.write("hello!");
|
||||||
|
}
|
||||||
|
|
||||||
|
mixin GenericMain!app
|
||||||
|
---
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
$(H3 Array elements)
|
||||||
|
|
||||||
|
|
||||||
In PHP, you can give a form element a name like `"something[]"`, and then
|
In PHP, you can give a form element a name like `"something[]"`, and then
|
||||||
`$_POST["something"]` gives an array. In D, you can use whatever name
|
`$_POST["something"]` gives an array. In D, you can use whatever name
|
||||||
you want, and access an array of values with the `cgi.getArray["name"]` and
|
you want, and access an array of values with the `cgi.getArray["name"]` and
|
||||||
`cgi.postArray["name"]` members.
|
`cgi.postArray["name"]` members.
|
||||||
|
|
||||||
```
|
$(H3 Databases)
|
||||||
echo("hello"); == cgi.write("hello");
|
|
||||||
|
|
||||||
$_SERVER["REMOTE_ADDR"] == cgi.remoteAddress
|
PHP has a lot of stuff in its standard library. cgi.d doesn't include most
|
||||||
$_SERVER["HTTP_HOST"] == cgi.host
|
of these, but the rest of my arsd repository has much of it. For example,
|
||||||
```
|
to access a MySQL database, download `database.d` and `mysql.d` from my
|
||||||
|
github repo, and try this code (assuming, of course, your database is
|
||||||
|
set up):
|
||||||
|
|
||||||
|
---
|
||||||
|
import arsd.cgi;
|
||||||
|
import arsd.mysql;
|
||||||
|
|
||||||
|
void app(Cgi cgi) {
|
||||||
|
auto database = new MySql("localhost", "username", "password", "database_name");
|
||||||
|
foreach(row; mysql.query("SELECT count(id) FROM people"))
|
||||||
|
cgi.write(row[0] ~ " people in database");
|
||||||
|
}
|
||||||
|
|
||||||
|
mixin GenericMain!app;
|
||||||
|
---
|
||||||
|
|
||||||
|
Similar modules are available for PostgreSQL, Microsoft SQL Server, and SQLite databases,
|
||||||
|
implementing the same basic interface.
|
||||||
|
|
||||||
See_Also:
|
See_Also:
|
||||||
|
|
||||||
You may also want to see dom.d, web.d, and html.d for more code for making
|
You may also want to see [arsd.dom], [arsd.web], and [arsd.html] for more code for making
|
||||||
web applications. database.d, mysql.d, postgres.d, and sqlite.d can help in
|
web applications.
|
||||||
|
|
||||||
|
For working with json, try [arsd.jsvar].
|
||||||
|
|
||||||
|
[arsd.database], [arsd.mysql], [arsd.postgres], [arsd.mssql], and [arsd.sqlite] can help in
|
||||||
accessing databases.
|
accessing databases.
|
||||||
|
|
||||||
If you are looking to access a web application via HTTP, try curl.d.
|
If you are looking to access a web application via HTTP, try [std.net.curl], [arsd.curl], or [arsd.http2].
|
||||||
|
|
||||||
Copyright:
|
Copyright:
|
||||||
|
|
||||||
cgi.d copyright 2008-2016, Adam D. Ruppe. Provided under the Boost Software License.
|
cgi.d copyright 2008-2018, Adam D. Ruppe. Provided under the Boost Software License.
|
||||||
|
|
||||||
Yes, this file is almost eight years old, and yes, it is still actively maintained and used.
|
Yes, this file is almost ten years old, and yes, it is still actively maintained and used.
|
||||||
+/
|
+/
|
||||||
module arsd.cgi;
|
module arsd.cgi;
|
||||||
|
|
||||||
|
|
71
mysql.d
71
mysql.d
|
@ -1,6 +1,14 @@
|
||||||
/// NOTE: If you're using MySQL client library v5.0 or less,
|
/++
|
||||||
/// you must pass this to dmd: -version=Less_Than_MySQL_51
|
Implementation of the `arsd.database.Database|database` interface for
|
||||||
/// This is important - otherwise you will see bizarre segfaults!
|
accessing MySQL (and MariaDB) databases. Uses the official MySQL client
|
||||||
|
library, and thus needs that installed to compile and run.
|
||||||
|
|
||||||
|
$(PITFALL
|
||||||
|
If you're using MySQL client library v5.0 or less,
|
||||||
|
you must pass this to dmd: `-version=Less_Than_MySQL_51`
|
||||||
|
This is important - otherwise you will see bizarre segfaults!
|
||||||
|
)
|
||||||
|
+/
|
||||||
module arsd.mysql;
|
module arsd.mysql;
|
||||||
|
|
||||||
|
|
||||||
|
@ -32,6 +40,14 @@ import std.conv;
|
||||||
import std.typecons;
|
import std.typecons;
|
||||||
import core.stdc.config;
|
import core.stdc.config;
|
||||||
|
|
||||||
|
/++
|
||||||
|
Represents a query result. You can loop over this with a
|
||||||
|
`foreach` statement to access individual [Row|rows].
|
||||||
|
|
||||||
|
[Row]s expose both an index and associative array interface,
|
||||||
|
so you can get `row[0]` for the first item, or `row["name"]`
|
||||||
|
to get a column by name from the result set.
|
||||||
|
+/
|
||||||
class MySqlResult : ResultSet {
|
class MySqlResult : ResultSet {
|
||||||
private int[string] mapping;
|
private int[string] mapping;
|
||||||
private MYSQL_RES* result;
|
private MYSQL_RES* result;
|
||||||
|
@ -72,20 +88,25 @@ class MySqlResult : ResultSet {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// The number of returned rows
|
||||||
override size_t length() {
|
override size_t length() {
|
||||||
if(result is null)
|
if(result is null)
|
||||||
return 0;
|
return 0;
|
||||||
return cast(int) mysql_num_rows(result);
|
return cast(int) mysql_num_rows(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Range primitive used by `foreach`
|
||||||
|
/// You may also use this to check if there was any result.
|
||||||
override bool empty() {
|
override bool empty() {
|
||||||
return itemsUsed == itemsTotal;
|
return itemsUsed == itemsTotal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Range primitive used by `foreach`
|
||||||
override Row front() {
|
override Row front() {
|
||||||
return row;
|
return row;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Range primitive used by `foreach`
|
||||||
override void popFront() {
|
override void popFront() {
|
||||||
itemsUsed++;
|
itemsUsed++;
|
||||||
if(itemsUsed < itemsTotal) {
|
if(itemsUsed < itemsTotal) {
|
||||||
|
@ -160,10 +181,22 @@ class MySqlResult : ResultSet {
|
||||||
Row row;
|
Row row;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
The main class for accessing the MySql database.
|
||||||
|
|
||||||
|
---
|
||||||
|
// connect to database with the constructor
|
||||||
|
auto db = new MySql("localhost", "my_user", "my_password", "my_database_name");
|
||||||
|
// use the query function to execute sql...
|
||||||
|
// use ? for data placeholders...
|
||||||
|
db.query("INSERT INTO people (id, name) VALUES (?, ?)", 10, "My Name");
|
||||||
|
// and use foreach to loop over result sets
|
||||||
|
foreach(row; db.query("SELECT id, name FROM people ORDER BY name LIMIT 10"))
|
||||||
|
writeln(row[0], " ", row["name"]); // index and name supported
|
||||||
|
---
|
||||||
|
+/
|
||||||
class MySql : Database {
|
class MySql : Database {
|
||||||
|
///
|
||||||
this(string host, string user, string pass, string db) {
|
this(string host, string user, string pass, string db) {
|
||||||
mysql = enforceEx!(DatabaseException)(
|
mysql = enforceEx!(DatabaseException)(
|
||||||
mysql_init(null),
|
mysql_init(null),
|
||||||
|
@ -182,6 +215,7 @@ class MySql : Database {
|
||||||
|
|
||||||
string dbname;
|
string dbname;
|
||||||
|
|
||||||
|
///
|
||||||
override void startTransaction() {
|
override void startTransaction() {
|
||||||
query("START TRANSACTION");
|
query("START TRANSACTION");
|
||||||
}
|
}
|
||||||
|
@ -198,12 +232,14 @@ class MySql : Database {
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
int lastInsertId() {
|
int lastInsertId() {
|
||||||
return cast(int) mysql_insert_id(mysql);
|
return cast(int) mysql_insert_id(mysql);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/// Builds and executes an INERT INTO statement
|
||||||
int insert(string table, MySqlResult result, string[string] columnsToModify, string[] columnsToSkip) {
|
int insert(string table, MySqlResult result, string[string] columnsToModify, string[] columnsToSkip) {
|
||||||
assert(!result.empty);
|
assert(!result.empty);
|
||||||
string sql = "INSERT INTO `" ~ table ~ "` ";
|
string sql = "INSERT INTO `" ~ table ~ "` ";
|
||||||
|
@ -308,23 +344,7 @@ class MySql : Database {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Gets a minimal ORM object from a query
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
ResultByDataObject!R queryDataObject(R = DataObject, T...)(string sql, T t) {
|
ResultByDataObject!R queryDataObject(R = DataObject, T...)(string sql, T t) {
|
||||||
// modify sql for the best data object grabbing
|
// modify sql for the best data object grabbing
|
||||||
sql = fixupSqlForDataObjectUse(sql);
|
sql = fixupSqlForDataObjectUse(sql);
|
||||||
|
@ -334,6 +354,7 @@ class MySql : Database {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// ditto
|
||||||
ResultByDataObject!R queryDataObjectWithCustomKeys(R = DataObject, T...)(string[string] keyMapping, string sql, T t) {
|
ResultByDataObject!R queryDataObjectWithCustomKeys(R = DataObject, T...)(string[string] keyMapping, string sql, T t) {
|
||||||
sql = fixupSqlForDataObjectUse(sql, keyMapping);
|
sql = fixupSqlForDataObjectUse(sql, keyMapping);
|
||||||
|
|
||||||
|
@ -343,9 +364,7 @@ class MySql : Database {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
///
|
||||||
|
|
||||||
|
|
||||||
int affectedRows() {
|
int affectedRows() {
|
||||||
return cast(int) mysql_affected_rows(mysql);
|
return cast(int) mysql_affected_rows(mysql);
|
||||||
}
|
}
|
||||||
|
@ -700,6 +719,7 @@ string fromCstring(cstring c, size_t len = size_t.max) {
|
||||||
|
|
||||||
|
|
||||||
// FIXME: this should work generically with all database types and them moved to database.d
|
// FIXME: this should work generically with all database types and them moved to database.d
|
||||||
|
///
|
||||||
Ret queryOneRow(Ret = Row, DB, string file = __FILE__, size_t line = __LINE__, T...)(DB db, string sql, T t) if(
|
Ret queryOneRow(Ret = Row, DB, string file = __FILE__, size_t line = __LINE__, T...)(DB db, string sql, T t) if(
|
||||||
(is(DB : Database))
|
(is(DB : Database))
|
||||||
// && (is(Ret == Row) || is(Ret : DataObject)))
|
// && (is(Ret == Row) || is(Ret : DataObject)))
|
||||||
|
@ -718,6 +738,7 @@ Ret queryOneRow(Ret = Row, DB, string file = __FILE__, size_t line = __LINE__, T
|
||||||
} else static assert(0, "Unsupported single row query return value, " ~ Ret.stringof);
|
} else static assert(0, "Unsupported single row query return value, " ~ Ret.stringof);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
class EmptyResultException : Exception {
|
class EmptyResultException : Exception {
|
||||||
this(string message, string file = __FILE__, size_t line = __LINE__) {
|
this(string message, string file = __FILE__, size_t line = __LINE__) {
|
||||||
super(message, file, line);
|
super(message, file, line);
|
||||||
|
|
1
script.d
1
script.d
|
@ -631,6 +631,7 @@ class TokenStream(TextStream) {
|
||||||
} else if(symbol == "/+") {
|
} else if(symbol == "/+") {
|
||||||
// FIXME: nesting comment
|
// FIXME: nesting comment
|
||||||
}
|
}
|
||||||
|
// FIXME: documentation comments
|
||||||
|
|
||||||
found = true;
|
found = true;
|
||||||
token.type = ScriptToken.Type.symbol;
|
token.type = ScriptToken.Type.symbol;
|
||||||
|
|
Loading…
Reference in New Issue