Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Golang, mysql: Error 1040: Too many connections

Tags:

mysql

go

I'm using the github.com/go-sql-driver/mysql driver for go.

I open a database:

db, err := sql.Open("mysql", str) 

Then I have two functions that are called 200 times each with following mysql code:

rows, err := db.Query("select name from beehives") if err != nil {     panic(err) }        defer rows.Close() 

The second:

    err = db.QueryRow("select id, secret, shortname from beehives where shortname = ?", beehive).Scan(&id, &secre     switch {     case err == sql.ErrNoRows:         err = errors.New("Beehive '"+beehive+"' not found.")     case err != nil:         panic("loginBeehive: "+ err.Error())     default:         // ... do the work 

The first one is panicing.

How can there be more than one connection when I open the database only once and how do I close them?

like image 647
Michael Avatar asked Jan 25 '15 10:01

Michael


People also ask

What causes MySQL too many connections?

The MySQL “Too many connections” error occurs when more queries are sent to a MySQL database than can be processed. The error can be fixed by setting a new number of maximum connections in the configuration file or globally.

How do I check for too many connections in MySQL?

If clients encounter Too many connections errors when attempting to connect to the mysqld server, all available connections are in use by other clients. The permitted number of connections is controlled by the max_connections system variable. To support more connections, set max_connections to a larger value.

How many MySQL connections can handle?

Simultaneous MySQL connection limits Each database user is limited to 38 simultaneous MySQL connections. This limitation helps to prevent overloading the MySQL server to the detriment of other sites hosted on the server.


2 Answers

sql.Open doesn't really open a connection to your database.

A sql.DB maintains a pool of connections to your database. Each time you query your database your program will try to get a connection from this pool or create a new one otherwise. These connections are than put back into the pool once you close them.

This is what rows.Close() does. Your db.QueryRow("...") does the same thing internally when you call Scan(...).

The basic problem is that you're creating too many queries, of which each one needs a connection, but you are not closing your connections fast enough. This way your program has to create a new connection for each query.

You can limit the maximum number of connections your program uses by calling SetMaxOpenConns on your sql.DB.

See http://go-database-sql.org/surprises.html for more information.

like image 112
nussjustin Avatar answered Sep 19 '22 23:09

nussjustin


The *DB object that you get back from sql.Open doesn't corresponds to a single connection. It is better thought as a handle for the database: it manages a connection pool for you.

You can control the number of open connections with `(*DB).SetMaxOpenConns and its pair for idling connections.

So basically what happens here is that db.Query and db.QueryRow tries to acquire a connection for themselves and the DB handle doesn't put any restrictions on the number of simultaneous connections so your code panics when it opens more than what mysql can handle.

like image 44
tmichel Avatar answered Sep 18 '22 23:09

tmichel