diff --git a/.eslintrc b/.eslintrc
index 65eb775f8de916b02d1779185c9627d1a26d635b..aa4c8e9f63613906e5ab6efead0a560c877894db 100644
--- a/.eslintrc
+++ b/.eslintrc
@@ -29,6 +29,7 @@
         "global-strict": 0,
         "new-cap": 0,
         "no-fallthrough": 0,
+        "no-useless-escape": 0,
         "no-case-declarations": 0,
         "react/no-is-mounted": 2,
         "react/prefer-es6-class": 2,
diff --git a/.github/ISSUE_TEMPLATE.md b/.github/ISSUE_TEMPLATE.md
index a8f484af147aca756f0f64c38b4d3ec0c00dda0e..614c0ce6f34c577fbfb4c3ae2722290e36309362 100644
--- a/.github/ISSUE_TEMPLATE.md
+++ b/.github/ISSUE_TEMPLATE.md
@@ -3,7 +3,7 @@ Before filing an issue we'd appreciate it if you could take a moment to ensure
 there isn't already an open issue or pull-request.
 -----
 
-If there's an exisiting issue, please add a :+1: reaction to the description of
+If there's an existing issue, please add a :+1: reaction to the description of
 the issue. One way we prioritize issues is by the number of :+1: reactions on
 their descriptions. Please DO NOT add `+1` or :+1: comments.
 
diff --git a/Dockerfile b/Dockerfile
index 9145b7b1d4630d129a047e9204f6545f8b5d1e61..12313c85fdbeddd68244c2c9e76f292268cc11c0 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -2,7 +2,7 @@
 # images hosted on Docker Hub https://hub.docker.com/r/metabase/metabase/ which use the
 # Dockerfile located at ./bin/docker/Dockerfile
 
-FROM java:openjdk-7-jre-alpine
+FROM java:openjdk-8-jre-alpine
 
 ENV JAVA_HOME=/usr/lib/jvm/default-jvm
 ENV PATH /usr/local/bin:$PATH
diff --git a/bin/aws-eb-docker/cloudformation-elasticbeanstalk.json.template b/bin/aws-eb-docker/cloudformation-elasticbeanstalk.json.template
new file mode 100644
index 0000000000000000000000000000000000000000..bc64e84b3e4871fad60e9aab3e129c614ab27b0b
--- /dev/null
+++ b/bin/aws-eb-docker/cloudformation-elasticbeanstalk.json.template
@@ -0,0 +1,149 @@
+{
+    "AWSTemplateFormatVersion": "2010-09-09",
+    "Description": "Deploys Metabase by creating an ElasticBeanstalk application in an contained in a CloudFormation stack.",
+    "Parameters": {
+        "MetabaseVersion": {
+            "Description": "Metabase Version",
+            "Type": "String",
+            "Default": "@@MB_TAG@@"
+        },
+        "KeyName": {
+            "Description": "Name of an existing EC2 KeyPair to enable SSH access to the AWS Elastic Beanstalk instance",
+            "Type": "AWS::EC2::KeyPair::KeyName"
+        },
+        "instanceType": {
+            "Description": "The Type of EC2 instance to use for running Metabase",
+            "Type": "String",
+            "Default": "t2.small"
+        },
+        "VPCid": {
+            "Description": "The VPC to use for running Metabase",
+            "Type": "AWS::EC2::VPC::Id"
+        },
+        "Subnets": {
+            "Description": "The VPC subnet(s) to use for running Metabase",
+            "Type": "List<AWS::EC2::Subnet::Id>"
+        }
+    },
+    "Resources": {
+        "metabaseApplication": {
+            "Type": "AWS::ElasticBeanstalk::Application",
+            "Properties": {
+                "Description": "Metabase Application",
+                "ApplicationVersions": [
+                    {
+                        "VersionLabel": {
+                            "Ref": "MetabaseVersion"
+                        },
+                        "Description": "Metabase Application Version",
+                        "SourceBundle": {
+                            "S3Bucket": "downloads.metabase.com",
+                            "S3Key": {
+                                "Fn::Join": [
+                                    "/",
+                                    [
+                                        {
+                                            "Ref": "MetabaseVersion"
+                                        },
+                                        "metabase-aws-eb.zip"
+                                    ]
+                                ]
+                            }
+                        }
+                    }
+                ],
+                "ConfigurationTemplates": [
+                    {
+                        "TemplateName": "MetabaseConfiguration",
+                        "Description": "Metabase Application Configuration",
+                        "SolutionStackName": "64bit Amazon Linux 2017.03 v2.7.0 running Docker 17.03.1-ce",
+                        "OptionSettings": [
+                            {
+                                "Namespace": "aws:autoscaling:launchconfiguration",
+                                "OptionName": "EC2KeyName",
+                                "Value": {
+                                    "Ref": "KeyName"
+                                }
+                            },
+                            {
+                                "Namespace": "aws:rds:dbinstance",
+                                "OptionName": "DBEngine",
+                                "Value": "postgres"
+                            },
+                            {
+                                "Namespace": "aws:rds:dbinstance",
+                                "OptionName": "DBAllocatedStorage",
+                                "Value": "10"
+                            },
+                            {
+                                "Namespace": "aws:rds:dbinstance",
+                                "OptionName": "DBInstanceClass",
+                                "Value": "db.t2.small"
+                            },
+                            {
+                                "Namespace": "aws:rds:dbinstance",
+                                "OptionName": "MultiAZDatabase",
+                                "Value": "false"
+                            },
+                            {
+                                "Namespace": "aws:rds:dbinstance",
+                                "OptionName": "DBDeletionPolicy",
+                                "Value": "Snapshot"
+                            },
+                            {
+                                "Namespace": "aws:autoscaling:launchconfiguration",
+                                "OptionName": "InstanceType",
+                                "Value": {
+                                    "Ref": "instanceType"
+                                }
+                            },
+                            {
+                                "Namespace": "aws:ec2:vpc",
+                                "OptionName": "VPCId",
+                                "Value": {
+                                    "Ref": "VPCid"
+                                }
+                            },
+                            {
+                                "Namespace": "aws:ec2:vpc",
+                                "OptionName": "Subnets",
+                                "Value": {
+                                    "Fn::Join": [
+                                        ",",
+                                        {
+                                            "Ref": "Subnets"
+                                        }
+                                    ]
+                                }
+                            }
+                        ]
+                    }
+                ]
+            }
+        },
+        "metabaseEnvironment": {
+            "Type": "AWS::ElasticBeanstalk::Environment",
+            "Properties": {
+                "ApplicationName": {
+                    "Ref": "metabaseApplication"
+                },
+                "Description": "AWS Elastic Beanstalk Environment for Metabase",
+                "TemplateName": "MetabaseConfiguration",
+                "VersionLabel": {
+                    "Ref": "MetabaseVersion"
+                }
+            }
+        }
+    },
+    "Outputs": {
+        "URL": {
+            "Description": "Metabase URL",
+            "Value": {
+                "Fn::GetAtt": [
+                    "metabaseEnvironment",
+                    "EndpointURL"
+                ]
+            }
+        }
+    }
+}
diff --git a/bin/aws-eb-docker/functions b/bin/aws-eb-docker/functions
index 70e776495087df0c7d2a68917230b2b1146f7af3..deb0bb26719ce04ef27ebde0367e45d91a66889d 100644
--- a/bin/aws-eb-docker/functions
+++ b/bin/aws-eb-docker/functions
@@ -12,6 +12,9 @@ export LANG=en_US.UTF-8
 export LANGUAGE=$LANG
 export LC_ALL=$LANG
 
+CF_TEMPLATE=cloudformation-elasticbeanstalk.json
+CF_TEMPLATE_PATH=/tmp/$CF_TEMPLATE
+S3_CF_TEMPLATE_PATH=s3://$ARTIFACTS_S3BUCKET/eb/$CF_TEMPLATE
 
 make_eb_version() {
     MB_TAG=$1
@@ -37,6 +40,9 @@ make_eb_version() {
     sed "s/@@MB_REPOSITORY@@/${MB_DOCKER_REPOSITORY}/" < ${BASEDIR}/Dockerrun.aws.json.template > ${BASEDIR}/Dockerrun.aws.json.tmp
     sed "s/@@MB_TAG@@/${MB_TAG}/" < ${BASEDIR}/Dockerrun.aws.json.tmp > ${BASEDIR}/Dockerrun.aws.json
 
+    # set the default version in the cloudformation template from the template-template (yo dawg i heard you like templates ;)
+    sed "s/@@MB_TAG@@/${MB_TAG}/" < ${BASEDIR}/$CF_TEMPLATE.template > $CF_TEMPLATE_PATH
+
     # create our EB zip file
     cd $BASEDIR; zip -r ${RELEASE_FILE} .ebextensions Dockerrun.aws.json; cd $CURRENTDIR
 
@@ -57,6 +63,10 @@ upload_eb_version() {
 
     echo "uploading /tmp/${MB_TAG}.zip -> $ARTIFACTS_S3BUCKET/eb/"
     aws s3 cp /tmp/${MB_TAG}.zip s3://$ARTIFACTS_S3BUCKET/eb/${MB_TAG}.zip
+
+    echo "uploading $CF_TEMPLATE_PATH -> $S3_CF_TEMPLATE_PATH"
+    aws s3 cp $CF_TEMPLATE_PATH $S3_CF_TEMPLATE_PATH
+
 }
 
 create_eb_version() {
diff --git a/bin/ci b/bin/ci
index 71f0748400d49527e2c1ddae35702b49c6666f01..a461187df7c94fd125108a2d60e655c452f1cc71 100755
--- a/bin/ci
+++ b/bin/ci
@@ -8,7 +8,7 @@ node-0() {
     if is_engine_enabled "mongo"; then
         run_step install-mongodb
     fi
-    run_step lein-test
+    MB_MYSQL_TEST_USER=ubuntu run_step lein-test
 }
 node-1() {
     is_enabled "drivers" && export ENGINES="h2,sqlserver,oracle" || export ENGINES="h2"
@@ -27,7 +27,7 @@ node-2() {
         run_step install-presto
     fi
     MB_ENCRYPTION_SECRET_KEY='Orw0AAyzkO/kPTLJRxiyKoBHXa/d6ZcO+p+gpZO/wSQ=' MB_DB_TYPE=mysql MB_DB_DBNAME=circle_test MB_DB_PORT=3306 MB_DB_USER=ubuntu MB_DB_HOST=localhost \
-        MB_PRESTO_HOST=localhost MB_PRESTO_PORT=8080 \
+        MB_PRESTO_TEST_HOST=localhost MB_PRESTO_TEST_PORT=8080 MB_POSTGRESQL_TEST_USER=ubuntu \
         run_step lein-test
 }
 node-3() {
@@ -55,13 +55,20 @@ node-6() {
     if is_enabled "jar" || is_enabled "e2e" || is_enabled "screenshots"; then
         run_step ./bin/build version frontend-fast sample-dataset uberjar
     fi
-    if is_enabled "e2e" || is_enabled "compare_screenshots"; then
-        USE_SAUCE=true \
-            run_step yarn run test-e2e
-    fi
-    if is_enabled "screenshots"; then
-        run_step node_modules/.bin/babel-node ./bin/compare-screenshots
+
+    # NOTE Atte Keinänen 6/23/17: Reuse the existing E2E infra for running integrated tests (they require the prebuild jar too)
+    if is_enabled "e2e"; then
+        run_step yarn run test-integrated
     fi
+
+    # TODO Atte Keinänen 6/22/17: Disabled due to Sauce problems, all tests will be converted to use Jest and Enzyme
+    # if is_enabled "e2e" || is_enabled "compare_screenshots"; then
+    #     USE_SAUCE=true \
+    #         run_step yarn run test-e2e
+    # fi
+    # if is_enabled "screenshots"; then
+    #     run_step node_modules/.bin/babel-node ./bin/compare-screenshots
+    # fi
 }
 
 
diff --git a/bin/docker/run_metabase.sh b/bin/docker/run_metabase.sh
index 93ef3fa837830f9486ba2db0f323f0bb0cae667e..cd2eb1c3d0da1955818c8917db6261759fbd6973 100755
--- a/bin/docker/run_metabase.sh
+++ b/bin/docker/run_metabase.sh
@@ -119,4 +119,4 @@ fi
 # Launch the application
 # exec is here twice on purpose to  ensure that metabase runs as PID 1 (the init process)
 # and thus receives signals sent to the container. This allows it to shutdown cleanly on exit
-exec su metabase -s /bin/sh -c "exec java $JAVA_OPTS -jar /app/metabase.jar"
+exec su metabase -s /bin/sh -c "exec java $JAVA_OPTS -jar /app/metabase.jar $@"
diff --git a/docs/administration-guide/03-metadata-editing.md b/docs/administration-guide/03-metadata-editing.md
index dae0d82a0c875637edb71bcde11ce04e36550cad..ecda1bad02a5280d93ad3fcf502ced2ebaec2676 100644
--- a/docs/administration-guide/03-metadata-editing.md
+++ b/docs/administration-guide/03-metadata-editing.md
@@ -74,6 +74,24 @@ This is also where you set mark special fields in a table:
 * Entity Name — different from the entity key, this is the field whose heading represents what each row in the table *is*. For example, in a Users table, the User column might be the entity name.
 * Foreign Key — this is a field in this table that uniquely identifies a *row* in another table. In other words, this is a field that, almost always, points to the primary key of another table. For example, in a Products table, you might have a Customer ID field that points to a Customers table, where Customer ID is the primary key.
 
+### Remapping field values
+One thing that happens commonly in tables is that you'll have a foreign key field, like `Product ID`, with a bunch of ID values in it, when what you actually want to see most of the time is the entity name, like the `Product Title`. You might also have fields which contain coded values that you'd prefer to show up as translated or readable values in your tables and charts — like changing `0`, `1`, and `2` to `Female`, `Male`, and `Other` for example.
+
+To do this in Metabase, click on the gear icon to the right of a field's Type dropdown in the Data Model section of the Admin Panel. You'll see a form with these options:
+
+![Remapping form](./images/remapping/form.png)
+
+`Visibility` and `Type` are the same as on the main Data Model page, but `Display values` lets you choose to swap out a field's values with something else.
+
+Foreign key remapping lets you swap out a foreign key's values with the values of any other field in the connected table. In this example, we're swapping out the `Product ID` field's values with the values in the `Title` field in the Product table:
+
+![Remapping form](./images/remapping/fk-mapping.png)
+
+Another option is custom remapping, which is currently only possible for numeric fields. This lets you map every number that occurs in this field to either a different numeric value or even to a text value, like in this example:
+
+![Remapping form](./images/remapping/custom-mapping.png)
+
+
 ---
 
 ## Next: managing users
diff --git a/docs/administration-guide/databases/vertica.md b/docs/administration-guide/databases/vertica.md
index f17768e780b14505af3d9a6c9b55aae8c05c48f7..a7b1491abccbbfa3ba50a51937b8fac40fe558ae 100644
--- a/docs/administration-guide/databases/vertica.md
+++ b/docs/administration-guide/databases/vertica.md
@@ -36,4 +36,5 @@ If you're running Metabase from the Mac App, the plugins directory defaults to `
 /Users/camsaul/Library/Application Support/Metabase/Plugins/vertica-jdbc-8.0.0-0.jar
 ```
 
-Finally, you can choose a custom plugins directory if the default doesn't suit your needs by setting the environment variable `MB_PLUGINS_DIR`.
+If you are running the Docker image or you want to use another directory for plugins, you should then specify a custom plugins directory by setting the environment variable `MB_PLUGINS_DIR`. 
+
diff --git a/docs/administration-guide/images/remapping/custom-mapping.png b/docs/administration-guide/images/remapping/custom-mapping.png
new file mode 100644
index 0000000000000000000000000000000000000000..149d59a19b5fc200574b1b2f559f27edf059983f
Binary files /dev/null and b/docs/administration-guide/images/remapping/custom-mapping.png differ
diff --git a/docs/administration-guide/images/remapping/fk-mapping.png b/docs/administration-guide/images/remapping/fk-mapping.png
new file mode 100644
index 0000000000000000000000000000000000000000..483c129b2bad062dd9525ebaaa922ddd41924558
Binary files /dev/null and b/docs/administration-guide/images/remapping/fk-mapping.png differ
diff --git a/docs/administration-guide/images/remapping/form.png b/docs/administration-guide/images/remapping/form.png
new file mode 100644
index 0000000000000000000000000000000000000000..4782d908d01a0d9b0b62a258f0956691aa4dd751
Binary files /dev/null and b/docs/administration-guide/images/remapping/form.png differ
diff --git a/docs/api-documentation.md b/docs/api-documentation.md
index 65de1da123cf064922ec7ea94982e8304d44186f..cc23fffd5a56d5be31d344cfaf6074e891da5654 100644
--- a/docs/api-documentation.md
+++ b/docs/api-documentation.md
@@ -1193,7 +1193,7 @@ Fetch the results for a Card in a publically-accessible Dashboard. Does not requ
 
 ## `GET /api/public/oembed`
 
-oEmbed endpoint used to retreive embed code and metadata for a (public) Metabase URL.
+oEmbed endpoint used to retrieve embed code and metadata for a (public) Metabase URL.
 
 ##### PARAMS:
 
@@ -1709,7 +1709,7 @@ Fetch the current `User`.
 
 ## `POST /api/user/`
 
-Create a new `User`, or or reäctivate an existing one.
+Create a new `User`, or or reactivate an existing one.
 
 You must be a superuser to do this.
 
@@ -1783,7 +1783,7 @@ You must be a superuser to do this.
 
 ## `GET /api/util/random_token`
 
-Return a cryptographically secure random 32-byte token, encoded as a hexidecimal string.
+Return a cryptographically secure random 32-byte token, encoded as a hexadecimal string.
    Intended for use when creating a value for `embedding-secret-key`.
 
 
@@ -1801,4 +1801,4 @@ Endpoint that checks if the supplied password meets the currently configured pas
 
 ##### PARAMS:
 
-*  **`password`** Insufficient password strength
\ No newline at end of file
+*  **`password`** Insufficient password strength
diff --git a/docs/operations-guide/running-metabase-on-docker.md b/docs/operations-guide/running-metabase-on-docker.md
index 0e2821f9df9bc3cb1e59d94fc8636bc5cb950270..940c430829e0f26c736bc848a631c63179e71145 100644
--- a/docs/operations-guide/running-metabase-on-docker.md
+++ b/docs/operations-guide/running-metabase-on-docker.md
@@ -2,6 +2,8 @@
 
 Metabase provides an official Docker image via Dockerhub that can be used for deployments on any system that is running Docker.
 
+If you're trying to upgrade your Metabase version on Docker, check out these [upgrading instructions](./start.md#upgrading-metabase).
+
 ### Launching Metabase on a new container
 
 Here's a quick one-liner to get you off the ground (please note, we recommend further configuration for production deployments below):
@@ -69,7 +71,35 @@ In this scenario all you need to do is make sure you launch Metabase with the co
 
 Keep in mind that Metabase will be connecting from within your docker container, so make sure that either you're using a fully qualified hostname or that you've set a proper entry in your container's `/etc/hosts file`.
 
-See instructions for [migrating from H2 to MySQL or Postgres](./start.md#migrating-from-using-the-h2-database-to-mysql-or-postgres).
+### Migrating from H2 to Postgres as the Metabase application database
+
+For general information, see instructions for [migrating from H2 to MySQL or Postgres](./start.md#migrating-from-using-the-h2-database-to-mysql-or-postgres).
+
+To migrate an existing Metabase container from an H2 application database to another database container (e.g. Postgres, MySQL), there are a few considerations to keep in mind:
+
+* The target database container must be accessible (i.e. on an available network)
+* The target database container must be supported (e.g. MySQL, Postgres)
+* The existing H2 database should be [mapped outside the running container](#mounting-a-mapped-file-storage-volume)
+
+The migration process involves 2 main steps:
+
+1. Stop the existing Metabase container
+2. Run a new, temporary Metabase container to perform the migration
+
+Using a Postgres container as the target, here's an example invocation:
+
+    docker run --name metabase-migration \
+        -v /path/metabase/data:/metabase-data \
+        -e "MB_DB_FILE=/metabase-data/metabase.db" \
+        -e "MB_DB_TYPE=postgres" \
+        -e "MB_DB_DBNAME=metabase" \
+        -e "MB_DB_PORT=5432" \
+        -e "MB_DB_USER=<username>" \
+        -e "MB_DB_PASS=<password>" \
+        -e "MB_DB_HOST=my-database-host" \
+        metabase/metabase load-from-h2
+
+To further explain the example: in addition to specifying the target database connection details, set the `MB_DB_FILE` environment variable for the source H2 database location, and pass the argument `load-from-h2` to begin migrating.
 
 ### Setting the Java Timezone
 
@@ -103,12 +133,12 @@ The DB contents will be left in a directory named metabase.db.
 Note that some older versions of metabase stored their db in a different default location.
 
     docker cp CONTAINER_ID:/metabase.db.mv.db metabase.db.mv.db
-    
+
 ### Fixing OutOfMemoryErrors in some hosted environments
 
 On some hosts Metabase can fail to start with an error message like:
 
     java.lang.OutOfMemoryError: Java heap space
-    
+
 If that happens, you'll need to set a JVM option to manually configure the maximum amount of memory the JVM uses for the heap. Refer
 to [these instructions](./start.md#metabase-fails-to-start-due-to-heap-space-outofmemoryerrors) for details on how to do that.
diff --git a/docs/operations-guide/start.md b/docs/operations-guide/start.md
index fbbf600b1ec094d4f09f3d06bd2e31fd40d8bcb5..1ad0822d8a3f1257c423ddd38d522538ffc23bbd 100644
--- a/docs/operations-guide/start.md
+++ b/docs/operations-guide/start.md
@@ -43,7 +43,7 @@ Community support only at this time, but we have reports of Metabase instances r
 
 # Upgrading Metabase
 
-Before you attempt to upgrade Metabase, you should make a backup of the database just in case. While it is unlikely you will need to rollback, it will do wonders for your peace of mind.
+Before you attempt to upgrade Metabase, you should make a backup of the application database just in case. While it is unlikely you will need to roll back, it will do wonders for your peace of mind.
 
 How you upgrade Metabase depends on how you are running it. See below for information on how to update Metabase on managed platforms.
 
@@ -51,14 +51,14 @@ How you upgrade Metabase depends on how you are running it. See below for inform
 
 
 #### Docker Image
-If you are running it via docker, then you simply kill the docker process, and start a new container with the latest image. On startup, Metabase will perform any upgrade tasks it needs to perform, and once it is finished, you'll be running the new version.
+If you are running Metabase via docker, then you simply need to kill the Docker process and start a new container with the latest Metabase image. On startup, Metabase will perform any upgrade tasks it needs to perform, and once it's finished you'll be running the new version.
 
 #### Jar file
-If you are running the JVM Jar file directly, then you simply kill the process, and restart the server. On startup, Metabase will perform any upgrade tasks it needs to perform, and once it is finished, you'll be running the new version.
+If you are running the JVM Jar file directly, then you simply kill the process and restart the server. On startup, Metabase will perform any upgrade tasks it needs to perform, and once it's finished you'll be running the new version.
 
 
-#### Mac OS X Application
-If you are using the Metabase app, you will be notified when there is a new version available. You will see a dialog displaying the changes in the latest version and prompt you to upgrade.
+#### macOS Application
+If you are using the Metabase macOS app, you will be notified when there is a new version available. You will see a dialog displaying the changes in the latest version and prompting you to upgrade.
 
 ![Autoupdate Confirmation Dialog](images/AutoupdateScreenshot.png)
 
@@ -113,7 +113,7 @@ You'll just need to set a JVM option to let it know explicitly how much memory i
 
     java -Xmx2g -jar metabase.jar
 
-Adjust this number as appropriate for your shared hosting instance. Make sure to set the number lower than the total amount of RAM available on your instance, because Metabase isn't the only process that'll be running. Generally, leaving 1-2 GB of RAM for these other processes should be enough; for example, you might set `-Xmx` to `1g` for an instance with 2 GB of RAM, `2g` for one with 4 GB of RAM, `6g` for an instance with 8 GB of RAM, and so forth. You may need to experment with these settings a bit to find the right number.
+Adjust this number as appropriate for your shared hosting instance. Make sure to set the number lower than the total amount of RAM available on your instance, because Metabase isn't the only process that'll be running. Generally, leaving 1-2 GB of RAM for these other processes should be enough; for example, you might set `-Xmx` to `1g` for an instance with 2 GB of RAM, `2g` for one with 4 GB of RAM, `6g` for an instance with 8 GB of RAM, and so forth. You may need to experiment with these settings a bit to find the right number.
 
 As above, you can use the environment variable `JAVA_TOOL_OPTIONS` to set JVM args instead of passing them directly to `java`. This is useful when running the Docker image,
 for example.
@@ -128,7 +128,7 @@ On Windows 10, if you see an error message like
 
     Exception in thread "main" java.lang.AssertionError: Assert failed: Unable to connect to Metabase DB.
 
-when running the JAR, you can unblock the file by right-clicking, clicking "Properties", and then clicking "Unblock".
+when running the JAR, you can unblock the file by right-clicking, clicking "Properties," and then clicking "Unblock."
 See Microsoft's documentation [here](https://blogs.msdn.microsoft.com/delay/p/unblockingdownloadedfile/) for more details on unblocking downloaded files.
 
 There are a few other reasons why Metabase might not be able to connect to your H2 DB. Metabase connects to the DB over a TCP port, and it's possible
diff --git a/docs/users-guide/04-asking-questions.md b/docs/users-guide/04-asking-questions.md
index 767576f96741d1ab034ac8498c8ff93d5f55c024..60f5725b26e495a119c4fa1965c4f5cb61c70c94 100644
--- a/docs/users-guide/04-asking-questions.md
+++ b/docs/users-guide/04-asking-questions.md
@@ -1,15 +1,23 @@
 
-## Asking Questions
+## Asking questions
 ---
-Metabase's two core concepts are questions and their corresponding answers. Everything else is based around questions and answers. To ask Metabase a question, click the New Question button at the top of the screen to go to the question builder.  
+Metabase's two core concepts are questions and their corresponding answers. Everything else is based around questions and answers. To ask Metabase a question, click the New Question button at the top of the screen to go to the question builder. (Note: to [create a new SQL query](04-asking-questions.html#using-sql), click the console icon in the top right of the new question screen.)
 
 ![queryinterfacebar](images/QueryInterfaceBar.png)
 
 Questions are made up of a number of parts: source data, filters, and answer output.
 
-### Source Data
+### Source data
 ---
-All of the data in databases are in tables. Typically, tables will be named for the thing that each row in the table contains. For example, in a Customers table, each row in the table would represent a single customer. This means that when you’re thinking about how to phrase your question, you’ll need to decide what your question is about, and which table has that information in it. The first dropdown menu in the question builder is where you’ll choose the table you want.  
+All of the data in databases are in tables. Typically, tables will be named for the thing that each row in the table contains. For example, in a Customers table, each row in the table would represent a single customer. This means that when you’re thinking about how to phrase your question, you’ll need to decide what your question is about, and which table has that information in it.
+
+The first dropdown menu in the question builder is where you’ll choose the database and table you want.
+
+#### Using saved questions as source data
+
+If you've [saved some questions](06-sharing-answers.html), in the Data menu you'll see the option to use one of your saved questions as source data. What this means in practice is that you can do things like use complex SQL queries to create new tables that can be used in a question just like any other table in your database.
+
+You can use any saved question as source data, provided you have [permission](../administration-guide/05-setting-permissions.html) to view that question. You can even use questions that were saved as a chart rather than a table. The only caveat is that you can't use a saved question which itself uses a saved question as source data. (That's more inception than Metabase can handle!)
 
 ### Filters
 ---
@@ -47,7 +55,7 @@ Now the relative date will be referencing the past 30 days from *today*, *not* f
 #### Using segments
 If your Metabase admins have created special named filters, called segments, for the table you’re viewing, they’ll appear at the top of the filter dropdown in purple text with a star next to them. These are shortcuts to sets of filters that are commonly used in your organization. They might be something like “Active Users,” or “Most Popular Products.”
 
-### Answer Output
+### Answer output
 ---
 The last section of the question builder is where you select what you want the output of your answer to be, under the View dropdown. You’re basically telling Metabase, “I want to view the…” Metabase can output the answer to your question in four different ways:
 
diff --git a/docs/users-guide/09-multi-series-charting.md b/docs/users-guide/09-multi-series-charting.md
index bec40797ed17296cd0682e75b142695c5b664429..f626a7d641e247ff534e6490deb208321614a3fb 100644
--- a/docs/users-guide/09-multi-series-charting.md
+++ b/docs/users-guide/09-multi-series-charting.md
@@ -30,7 +30,7 @@ If you already have two or more saved questions you’d like to compare, and the
 
 ![multiseriesmodal1](images/MultiSeriesModal1.png)
 
-The X and Y axis will automagically update if necessary and Metabase will create a legend using the existing card titles to help you understand which question maps to which series on the chart. Repeat this process as many times as you need.
+The X and Y axis will automatically update if necessary and Metabase will create a legend using the existing card titles to help you understand which question maps to which series on the chart. Repeat this process as many times as you need.
 
 To remove a series either uncheck the box, or click the x next to the title in the legend above the chart.
 
diff --git a/docs/users-guide/11-metabot.md b/docs/users-guide/11-metabot.md
index f1100861552b2c63d0d92db39de939e4a3bef973..bb58fbdeb55fd34c350a039f6970a9821d7768f0 100644
--- a/docs/users-guide/11-metabot.md
+++ b/docs/users-guide/11-metabot.md
@@ -1,43 +1,45 @@
 ## Getting answers in Slack with MetaBot
 
-You can already send data to Slack on  a set schedule via [Pulses](10-pulses.md) but what about when you need an answer right now? Say hello to MetaBot.
+You can already send data to Slack on a set schedule with [Pulses](10-pulses.md), but what about when you need an answer right now? Say hello to MetaBot.
 
 MetaBot helps add context to conversations you’re having in Slack by letting you insert results from Metabase.
 
 ### Connecting to Slack.
 To use MetaBot with Slack you’ll first need to connect Metabase to your Slack with an API token.
 
-See [Setting up Slack](http://www.metabase.com/docs/latest/administration-guide/07-setting-up-slack) for more information.
+See [Setting up Slack](../administration-guide/09-setting-up-slack.md) for more information.
 
 
 ### What can MetaBot do?
-
 MetaBot can show individual questions and also lists of questions that have already been asked in Metabase.
 
 If you ever need help remembering what MetaBot can do, just type ```metabot help``` in Slack.
 
 ![MetaBot help](images/metabot/MetabotHelp.png)
 
+### Where can I use MetaBot?
+You can talk to MetaBot in any Slack channel, including private ones, as long as you've invited the MetaBot Slack user to that channel.
+
 ### Showing questions
 
 To see a question from Metabase in Slack type
-'''metabot show "<question-name>"''' where question name is the title of one of saved questions. If you have several similarly named questions Metabot will ask you to differentiate between the two by typing the number next to the name.
+```metabot show [question name]``` where ```question name``` is the title of one of your saved questions. If you have several similarly named questions, Metabot will ask you to differentiate between the two by typing the number next to the name.
 
 ![MetaBot similar](images/metabot/MetabotSimilarItems.png)
 
-That number is the ID number of the question in Metabase and if you find yourself using the same question over and over again you can save a bit of time by typing “MetaBot show 19.”
+That number is the ID number of the question in Metabase, and if you find yourself using the same question over and over again you can save a bit of time by typing “MetaBot show 19.”
 
 ![MetaBot show](images/metabot/MetabotShow.png)
 
 ## Listing questions
-If you don’t have a sense of which questions you want to view in  Slack, you can type ```MetaBot list``` to get a list of recent questions from your Metabase.
+If you don’t have a sense of which questions you want to view in  Slack, you can type ```MetaBot list``` to get a list of the most recently saved questions in your Metabase.
 
 ![MetaBot show](images/metabot/MetabotList.png)
 
 
 ## To review
 
-- [Connect to Slack](10-pulses.md) to start using MetaBot.
+- [Connect to Slack](../administration-guide/09-setting-up-slack.md) to start using MetaBot.
 - Show data from Metabase in Slack using ```metabot show <question-id>```
 - Search for questions by typing ```metabot show <search-term>```
 - Get a list of questions by typing ```metabot list```
diff --git a/frontend/interfaces/underscore.js b/frontend/interfaces/underscore.js
index d5c7251adc744a4f6fd045eba1a393d6b20327a2..8b3511ac04ba9054b193152dff50dbb5dc1a3333 100644
--- a/frontend/interfaces/underscore.js
+++ b/frontend/interfaces/underscore.js
@@ -56,6 +56,7 @@ declare module "underscore" {
   declare function pick(o: {[key: any]: any}, ...properties: string[]): {[key: any]: any};
   declare function pick(o: {[key: any]: any}, predicate: (val: any, key: any, object: {[key: any]: any})=>boolean): {[key: any]: any};
   declare function pluck(o: Array<{[key: any]: any}>, propertyNames: string): Array<any>;
+  declare function has(o: {[key: any]: any}, ...properties: string[]): boolean;
 
   declare function difference<T>(array: T[], ...others: T[][]): T[];
 
diff --git a/frontend/src/metabase-lib/lib/Dimension.js b/frontend/src/metabase-lib/lib/Dimension.js
index 37af20f62a411978fb4f586e44446ddbd2f7a421..b8dd64b6f371108fe3448f3b8bffacba44dd28c4 100644
--- a/frontend/src/metabase-lib/lib/Dimension.js
+++ b/frontend/src/metabase-lib/lib/Dimension.js
@@ -32,7 +32,7 @@ type DimensionOption = {
 /**
  * Dimension base class, represents an MBQL field reference.
  *
- * Used for displaying fields (like Created At) and their "sub-dimensions" (like Hour of day)
+ * Used for displaying fields (like Created At) and their "sub-dimensions" (like Created At by Day)
  * in field lists and active value widgets for filters, aggregations and breakouts.
  *
  * @abstract
@@ -338,7 +338,8 @@ export class FieldIDDimension extends FieldDimension {
     }
 
     field() {
-        return this._metadata.fields[this._args[0]] || new Field();
+        return (this._metadata && this._metadata.fields[this._args[0]]) ||
+            new Field();
     }
 }
 
@@ -375,7 +376,8 @@ export class FKDimension extends FieldDimension {
     }
 
     field() {
-        return this._metadata.fields[this._args[0]] || new Field();
+        return (this._metadata && this._metadata.fields[this._args[0]]) ||
+            new Field();
     }
 
     render() {
diff --git a/frontend/src/metabase-lib/lib/Mode.spec.js b/frontend/src/metabase-lib/lib/Mode.spec.js
index 07b6685b065416f6812b18e8bbc3ef709da51f73..7591a8d6f9ca6e08251903993673d9d336d4cf31 100644
--- a/frontend/src/metabase-lib/lib/Mode.spec.js
+++ b/frontend/src/metabase-lib/lib/Mode.spec.js
@@ -56,39 +56,31 @@ describe("Mode", () => {
 
     describe("actions()", () => {
         describe("for a new question with Orders table and Raw data aggregation", () => {
+            pending();
             it("returns a correct number of mode actions", () => {
-                expect(rawDataQuestionMode.actions().length).toBe(4);
+                expect(rawDataQuestionMode.actions().length).toBe(3);
             });
-            it("returns 'View this as a table' as mode action 1", () => {
+            it("returns a defined metric as mode action 1", () => {
                 expect(rawDataQuestionMode.actions()[0].name).toBe(
-                    "underlying-data"
-                );
-                expect(rawDataQuestionMode.actions()[0].icon).toBe("table");
-                expect(rawDataQuestionMode.actions()[0].title).toBe(
-                    "View this as a table"
-                );
-            });
-            it("returns a defined metric as mode action 2", () => {
-                expect(rawDataQuestionMode.actions()[1].name).toBe(
                     "common-metric"
                 );
                 // TODO: Sameer 6/16/17
                 // This is wack and not really testable. We shouldn't be passing around react components in this imo
                 // expect(question.actions()[1].title.props.children).toBe("Total Order Value");
             });
-            it("returns a count timeseries as mode action 3", () => {
-                expect(rawDataQuestionMode.actions()[2].name).toBe(
+            it("returns a count timeseries as mode action 2", () => {
+                expect(rawDataQuestionMode.actions()[1].name).toBe(
                     "count-by-time"
                 );
-                expect(rawDataQuestionMode.actions()[2].icon).toBe("line");
+                expect(rawDataQuestionMode.actions()[1].icon).toBe("line");
                 // TODO: Sameer 6/16/17
                 // This is wack and not really testable. We shouldn't be passing around react components in this imo
                 // expect(question.actions()[2].title.props.children).toBe("Count of rows by time");
             });
-            it("returns summarize as mode action 4", () => {
-                expect(rawDataQuestionMode.actions()[3].name).toBe("summarize");
-                expect(rawDataQuestionMode.actions()[3].icon).toBe("sum");
-                expect(rawDataQuestionMode.actions()[3].title).toBe(
+            it("returns summarize as mode action 3", () => {
+                expect(rawDataQuestionMode.actions()[2].name).toBe("summarize");
+                expect(rawDataQuestionMode.actions()[2].icon).toBe("sum");
+                expect(rawDataQuestionMode.actions()[2].title).toBe(
                     "Summarize this segment"
                 );
             });
diff --git a/frontend/src/metabase-lib/lib/Question.js b/frontend/src/metabase-lib/lib/Question.js
index 257573e7902666ee9ce80caba22c3969c37aa820..c4fa66fbc4f50de784e7efe1c0885594ef54760e 100644
--- a/frontend/src/metabase-lib/lib/Question.js
+++ b/frontend/src/metabase-lib/lib/Question.js
@@ -278,6 +278,10 @@ export default class Question {
         return this._card && this._card.name;
     }
 
+    setDisplayName(name: String) {
+        return this.setCard(assoc(this.card(), "name", name));
+    }
+
     id(): number {
         return this._card && this._card.id;
     }
@@ -308,17 +312,21 @@ export default class Question {
     async getResults(
         { cancelDeferred, isDirty = false, ignoreCache = false } = {}
     ): Promise<[Dataset]> {
+        // TODO Atte Keinänen 7/5/17: Should we clean this query with Query.cleanQuery(query) before executing it?
+
         const canUseCardApiEndpoint = !isDirty && this.isSaved();
 
-        const parametersList = this.parametersList().map(param =>
-            _.pick(param, "target", "type", "value"));
-        const hasParameters = parametersList.length > 0;
+        const parameters = this.parametersList()
+            // include only parameters that have a value applied
+            .filter(param => _.has(param, "value"))
+            // only the superset of parameters object that API expects
+            .map(param => _.pick(param, "type", "target", "value"));
 
         if (canUseCardApiEndpoint) {
             const queryParams = {
                 cardId: this.id(),
                 ignore_cache: ignoreCache,
-                ...(hasParameters ? { parameters: parametersList } : {})
+                parameters
             };
 
             return [
@@ -330,7 +338,7 @@ export default class Question {
             const getDatasetQueryResult = datasetQuery => {
                 const datasetQueryWithParameters = {
                     ...datasetQuery,
-                    ...(hasParameters ? { parameters: parametersList } : {})
+                    parameters
                 };
 
                 return MetabaseApi.dataset(
@@ -384,7 +392,9 @@ export default class Question {
                 return false;
             }
         } else {
-            const origCardSerialized = originalQuestion._serializeForUrl();
+            const origCardSerialized = originalQuestion._serializeForUrl({
+                includeOriginalCardId: false
+            });
             const currentCardSerialized = this._serializeForUrl({
                 includeOriginalCardId: false
             });
diff --git a/frontend/src/metabase-lib/lib/Question.spec.js b/frontend/src/metabase-lib/lib/Question.spec.js
index b50c571091e15267c50ab776fea1aae90de5bd6a..0a47c29840c57f5d44c3f7512ed8fc7245045958 100644
--- a/frontend/src/metabase-lib/lib/Question.spec.js
+++ b/frontend/src/metabase-lib/lib/Question.spec.js
@@ -35,7 +35,7 @@ describe("Question", () => {
                 expect(question.canRun()).toBe(true);
             });
             it("has correct display settings", () => {
-                expect(question.display()).toBeUndefined();
+                expect(question.display()).toBe("table");
             });
             it("has correct mode", () => {
                 expect(question.mode().name()).toBe("segment");
@@ -162,6 +162,24 @@ describe("Question", () => {
         });
     });
 
+    describe("CARD METHODS", () => {
+        describe("card()", () => {
+            it("A question wraps a query/card and you can see the underlying card with card()", () => {
+                const question = new Question(metadata, orders_raw_card);
+                expect(question.card()).toEqual(orders_raw_card);
+            });
+        });
+
+        describe("setCard(card)", () => {
+            it("changes the underlying card", () => {
+                const question = new Question(metadata, orders_raw_card);
+                expect(question.card()).toEqual(orders_raw_card);
+                const newQustion = question.setCard(orders_count_by_id_card);
+                expect(question.card()).toEqual(orders_raw_card);
+                expect(newQustion.card()).toEqual(orders_count_by_id_card);
+            });
+        });
+    });
     describe("RESETTING METHODS", () => {
         describe("withoutNameAndId()", () => {
             it("unsets the name and id", () => {
@@ -535,7 +553,16 @@ describe("Question", () => {
         });
     });
 
-    fdescribe("COMPARISON TO OTHER QUESTIONS", () => {
+    describe("QUESTION EXECUTION", () => {
+        describe("getResults()", () => {
+            it("executes correctly a native query with field filter parameters", () => {
+                pending();
+                // test also here a combo of parameter with a value + parameter without a value + parameter with a default value
+            });
+        });
+    });
+
+    describe("COMPARISON TO OTHER QUESTIONS", () => {
         describe("isDirtyComparedTo(question)", () => {
             it("New questions are automatically dirty", () => {
                 const question = new Question(metadata, orders_raw_card);
@@ -559,7 +586,7 @@ describe("Question", () => {
             it("returns a question with hash for an unsaved question", () => {
                 const question = new Question(metadata, orders_raw_card);
                 expect(question.getUrl()).toBe(
-                    "/question#eyJuYW1lIjoiUmF3IG9yZGVycyBkYXRhIiwiZGF0YXNldF9xdWVyeSI6eyJ0eXBlIjoicXVlcnkiLCJkYXRhYmFzZSI6MSwicXVlcnkiOnsic291cmNlX3RhYmxlIjoxfX19"
+                    "/question#eyJuYW1lIjoiUmF3IG9yZGVycyBkYXRhIiwiZGF0YXNldF9xdWVyeSI6eyJ0eXBlIjoicXVlcnkiLCJkYXRhYmFzZSI6MSwicXVlcnkiOnsic291cmNlX3RhYmxlIjoxfX0sImRpc3BsYXkiOiJ0YWJsZSIsInZpc3VhbGl6YXRpb25fc2V0dGluZ3MiOnt9fQ=="
                 );
             });
         });
diff --git a/frontend/src/metabase-lib/lib/metadata/Field.js b/frontend/src/metabase-lib/lib/metadata/Field.js
index efdf769982cc197e43e731cca262a68ac166db23..5ae5e0fd0ab99191e5c733519024b32a9d2ab449 100644
--- a/frontend/src/metabase-lib/lib/metadata/Field.js
+++ b/frontend/src/metabase-lib/lib/metadata/Field.js
@@ -105,4 +105,20 @@ export default class Field extends Base {
             return this.operators_lookup[op];
         }
     }
+
+    /**
+     * Returns a default breakout MBQL clause for this field
+     *
+     * Tries to look up a default subdimension (like "Created At: Day" for "Created At" field)
+     * and if it isn't found, uses the plain field id dimension (like "Product ID") as a fallback.
+     */
+    getDefaultBreakout = () => {
+        const fieldIdDimension = this.dimension();
+        const defaultSubDimension = fieldIdDimension.defaultDimension();
+        if (defaultSubDimension) {
+            return defaultSubDimension.mbql();
+        } else {
+            return fieldIdDimension.mbql();
+        }
+    };
 }
diff --git a/frontend/src/metabase-lib/lib/metadata/Table.spec.js b/frontend/src/metabase-lib/lib/metadata/Table.spec.js
index 72096dd81d475390932576e8e983056d4b90256b..c5576bb73a2383da2e19e4f9f979549394e145ce 100644
--- a/frontend/src/metabase-lib/lib/metadata/Table.spec.js
+++ b/frontend/src/metabase-lib/lib/metadata/Table.spec.js
@@ -24,12 +24,10 @@ describe("Table", () => {
     });
 
     describe("dimensions", () => {
-        /*
-        it('returns dimension fields', () => {
-            console.log(metadata)
-            expect(table.dimensions().length)
-        })
-        */
+        it("returns dimension fields", () => {
+            pending();
+            // expect(table.dimensions().length)
+        });
     });
 
     describe("date fields", () => {
diff --git a/frontend/src/metabase-lib/lib/queries/NativeQuery.spec.js b/frontend/src/metabase-lib/lib/queries/NativeQuery.spec.js
index f27098c5c1b88e6351146a40242d26b24fc84b70..b702964db51d7c504d8a515f3db0ebef15c355f3 100644
--- a/frontend/src/metabase-lib/lib/queries/NativeQuery.spec.js
+++ b/frontend/src/metabase-lib/lib/queries/NativeQuery.spec.js
@@ -160,7 +160,6 @@ describe("NativeQuery", () => {
                 const newQuery = makeQuery().updateQueryText(
                     "SELECT * from ORDERS where total < {{max_price}}"
                 );
-                console.log(newQuery.templateTags());
                 expect(newQuery.templateTags().length).toBe(1);
             });
         });
diff --git a/frontend/src/metabase/__support__/integrated_tests.js b/frontend/src/metabase/__support__/integrated_tests.js
index 9ae186f1facc79fd2c78c7f92ca4b0b77780f093..9a812221c2b1edfb48a3a97672dc9e31643a2c5b 100644
--- a/frontend/src/metabase/__support__/integrated_tests.js
+++ b/frontend/src/metabase/__support__/integrated_tests.js
@@ -4,8 +4,13 @@
  * Import this file before other imports in integrated tests
  */
 
+// Mocks in a separate file as they would clutter this file
+// This must be before all other imports
+import "./integrated_tests_mocks";
+
+import { format as urlFormat } from "url";
 import api from "metabase/lib/api";
-import { SessionApi } from "metabase/services";
+import { CardApi, SessionApi } from "metabase/services";
 import { METABASE_SESSION_COOKIE } from "metabase/lib/cookies";
 import reducers from 'metabase/reducers-main';
 
@@ -14,27 +19,58 @@ import { Provider } from 'react-redux';
 
 import { createMemoryHistory } from 'history'
 import { getStore } from "metabase/store";
-import { useRouterHistory } from "react-router";
+import { createRoutes, Router, useRouterHistory } from "react-router";
+import _ from 'underscore';
 
 // Importing isomorphic-fetch sets the global `fetch` and `Headers` objects that are used here
 import fetch from 'isomorphic-fetch';
 
-// Mocks in a separate file as they would clutter this file
-import "./integrated_tests_mocks";
+import { refreshSiteSettings } from "metabase/redux/settings";
+import { getRoutes } from "metabase/routes";
 
-// Stores the current login session
-var loginSession = null;
+let hasCreatedStore = false;
+let loginSession = null; // Stores the current login session
+let simulateOfflineMode = false;
 
 /**
  * Login to the Metabase test instance with default credentials
  */
 export async function login() {
-    loginSession = await SessionApi.create({ email: "bob@metabase.com", password: "12341234"});
+    if (hasCreatedStore) {
+        console.warn(
+            "Warning: You have created a test store before calling login() which means that up-to-date site settings " +
+            "won't be in the store unless you call `refreshSiteSettings` action manually. Please prefer " +
+            "logging in before all tests and creating the store inside an individual test or describe block."
+        )
+    }
+
+    if (process.env.SHARED_LOGIN_SESSION_ID) {
+        loginSession = { id: process.env.SHARED_LOGIN_SESSION_ID }
+    } else {
+        loginSession = await SessionApi.create({ username: "bob@metabase.com", password: "12341234"});
+    }
+}
+
+/**
+ * Calls the provided function while simulating that the browser is offline.
+ */
+export async function whenOffline(callWhenOffline) {
+    simulateOfflineMode = true;
+    return callWhenOffline()
+        .then((result) => {
+            simulateOfflineMode = false;
+            return result;
+        })
+        .catch((e) => {
+            simulateOfflineMode = false;
+            throw e;
+        });
 }
 
+
 // Patches the metabase/lib/api module so that all API queries contain the login credential cookie.
 // Needed because we are not in a real web browser environment.
-api._makeRequest = async (method, url, headers, body, data, options) => {
+api._makeRequest = async (method, url, headers, requestBody, data, options) => {
     const headersWithSessionCookie = {
         ...headers,
         ...(loginSession ? {"Cookie": `${METABASE_SESSION_COOKIE}=${loginSession.id}`} : {})
@@ -44,21 +80,42 @@ api._makeRequest = async (method, url, headers, body, data, options) => {
         credentials: "include",
         method,
         headers: new Headers(headersWithSessionCookie),
-        ...(body ? {body} : {})
+        ...(requestBody ? { body: requestBody } : {})
     };
 
-    const result = await fetch(api.basename + url, fetchOptions);
+    let isCancelled = false
+    if (options.cancelled) {
+        options.cancelled.then(() => {
+            isCancelled = true;
+        });
+    }
+    const result = simulateOfflineMode
+        ? { status: 0, responseText: '' }
+        : (await fetch(api.basename + url, fetchOptions));
+
+    if (isCancelled) {
+        throw { status: 0, data: '', isCancelled: true}
+    }
+
+    let resultBody = null
+    try {
+        resultBody = await result.text();
+        // Even if the result conversion to JSON fails, we still return the original text
+        // This is 1-to-1 with the real _makeRequest implementation
+        resultBody = JSON.parse(resultBody);
+    } catch (e) {}
+
+
     if (result.status >= 200 && result.status <= 299) {
-        try {
-            return await result.json();
-        } catch (e) {
-            return null;
-        }
+        return resultBody
     } else {
-        const error = {status: result.status, data: await result.json()}
-        console.log('A request made in a test failed with the following error:')
-        console.dir(error, { depth: null })
-
+        const error = { status: result.status, data: resultBody, isCancelled: false }
+        if (!simulateOfflineMode) {
+            console.log('A request made in a test failed with the following error:');
+            console.log(error, { depth: null });
+            console.log(`The original request: ${method} ${url}`);
+            if (requestBody) console.log(`Original payload: ${requestBody}`);
+        }
         throw error
     }
 }
@@ -73,35 +130,141 @@ if (process.env.E2E_HOST) {
     process.quit(0)
 }
 
-export const createReduxStore = () => {
-    return getStore(reducers);
-}
-export const createReduxStoreWithBrowserHistory = () => {
+/**
+ * Creates an augmented Redux store for testing the whole app including browser history manipulation. Includes:
+ * - A simulated browser history that is used by react-router
+ * - Methods for
+ *     * manipulating the browser history
+ *     * waiting until specific Redux actions have been dispatched
+ *     * getting a React container subtree for the current route
+ */
+
+export const createTestStore = async () => {
+    hasCreatedStore = true;
+
     const history = useRouterHistory(createMemoryHistory)();
-    const store = getStore(reducers, history);
-    return { history, store }
+    const store = getStore(reducers, history, undefined, (createStore) => testStoreEnhancer(createStore, history));
+    store.setFinalStoreInstance(store);
+
+    await store.dispatch(refreshSiteSettings());
+    return store;
 }
 
-/**
- * Returns the given React container with an access to a global Redux store
- */
-export function linkContainerToGlobalReduxStore(component) {
-    return (
-        <Provider store={globalReduxStore}>
-            {component}
-        </Provider>
-    );
+const testStoreEnhancer = (createStore, history) => {
+    return (...args) => {
+        const store = createStore(...args);
+
+        const testStoreExtensions = {
+            _originalDispatch: store.dispatch,
+            _onActionDispatched: null,
+            _dispatchedActions: [],
+            _finalStoreInstance: null,
+
+            setFinalStoreInstance: (finalStore) => {
+                store._finalStoreInstance = finalStore;
+            },
+
+            dispatch: (action) => {
+                const result = store._originalDispatch(action);
+                store._dispatchedActions = store._dispatchedActions.concat([action]);
+                if (store._onActionDispatched) store._onActionDispatched();
+                return result;
+            },
+
+            resetDispatchedActions: () => {
+                store._dispatchedActions = [];
+            },
+
+            /**
+             * Waits until all actions with given type identifiers have been called or fails if the maximum waiting
+             * time defined in `timeout` is exceeded.
+             *
+             * Convenient in tests for waiting specific actions to be executed after mounting a React container.
+             */
+            waitForActions: (actionTypes, {timeout = 8000} = {}) => {
+                actionTypes = Array.isArray(actionTypes) ? actionTypes : [actionTypes]
+
+                const allActionsAreTriggered = () => _.every(actionTypes, actionType =>
+                    store._dispatchedActions.filter((action) => action.type === actionType).length > 0
+                );
+
+                if (allActionsAreTriggered()) {
+                    // Short-circuit if all action types are already in the history of dispatched actions
+                    return;
+                } else {
+                    return new Promise((resolve, reject) => {
+                        store._onActionDispatched = () => {
+                            if (allActionsAreTriggered()) resolve()
+                        };
+                        setTimeout(() => {
+                            store._onActionDispatched = null;
+
+                            if (allActionsAreTriggered()) {
+                                // TODO: Figure out why we sometimes end up here instead of _onActionDispatched hook
+                                resolve()
+                            } else {
+                                return reject(
+                                    new Error(
+                                        `Actions ${actionTypes.join(", ")} were not dispatched within ${timeout}ms. ` +
+                                        `Dispatched actions so far: ${store._dispatchedActions.map((a) => a.type).join(", ")}`
+                                    )
+                                )
+                            }
+
+                        }, timeout)
+                    });
+                }
+            },
+
+            getDispatchedActions: () => {
+                return store._dispatchedActions;
+            },
+
+            pushPath: (path) => history.push(path),
+            goBack: () => history.goBack(),
+            getPath: () => urlFormat(history.getCurrentLocation()),
+
+            connectContainer: (reactContainer) => {
+                const routes = createRoutes(getRoutes(store._finalStoreInstance))
+                return store._connectWithStore(
+                    <Router
+                        routes={routes}
+                        history={history}
+                        render={(props) => React.cloneElement(reactContainer, props)}
+                    />
+                );
+            },
+
+            getAppContainer: () => {
+                return store._connectWithStore(
+                    <Router history={history}>
+                        {getRoutes(store._finalStoreInstance)}
+                    </Router>
+                )
+            },
+
+            // eslint-disable-next-line react/display-name
+            _connectWithStore: (reactContainer) =>
+                <Provider store={store._finalStoreInstance}>
+                    {reactContainer}
+                </Provider>
+
+        }
+
+        return Object.assign(store, testStoreExtensions);
+    }
 }
 
-/**
- * A Redux store that is shared between subsequent tests,
- * intended to reduce the need for reloading metadata between every test
- */
-const {
-    history: globalBrowserHistory,
-    store: globalReduxStore
-} = createReduxStoreWithBrowserHistory()
-export { globalBrowserHistory, globalReduxStore }
+export const clickRouterLink = (linkEnzymeWrapper) =>
+    linkEnzymeWrapper.simulate('click', { button: 0 });
 
-jasmine.DEFAULT_TIMEOUT_INTERVAL = 10000;
+// Commonly used question helpers that are temporarily here
+// TODO Atte Keinänen 6/27/17: Put all metabase-lib -related test helpers to one file
+export const createSavedQuestion = async (unsavedQuestion) => {
+    const savedCard = await CardApi.create(unsavedQuestion.card())
+    const savedQuestion = unsavedQuestion.setCard(savedCard);
+    savedQuestion._card = { ...savedQuestion._card, original_card_id: savedQuestion.id() }
+    return savedQuestion
+}
 
+jasmine.DEFAULT_TIMEOUT_INTERVAL = 20000;
diff --git a/frontend/src/metabase/__support__/integrated_tests_mocks.js b/frontend/src/metabase/__support__/integrated_tests_mocks.js
index f8c5046d44bd20c779585cce7d647f96e1da9a43..9cb0446392261cfdd8580dc90d8430deb72bda97 100644
--- a/frontend/src/metabase/__support__/integrated_tests_mocks.js
+++ b/frontend/src/metabase/__support__/integrated_tests_mocks.js
@@ -25,3 +25,12 @@ jest.mock("ace/snippets/sqlserver", () => {}, {virtual: true});
 jest.mock("ace/snippets/json", () => {}, {virtual: true});
 jest.mock("ace/snippets/json", () => {}, {virtual: true});
 jest.mock("ace/ext-language_tools", () => {}, {virtual: true});
+
+import * as modal from "metabase/components/Modal";
+modal.default = modal.TestModal;
+
+import * as tooltip from "metabase/components/Tooltip";
+tooltip.default = tooltip.TestTooltip
+
+import * as popover from "metabase/components/Popover";
+popover.default = popover.TestPopover
diff --git a/frontend/src/metabase/__support__/sample_dataset_fixture.js b/frontend/src/metabase/__support__/sample_dataset_fixture.js
index 38c9a486104ddc6507a74ade743f20435c82b475..7e67ebd7519da37e7c8e7fbf00df30efb50533f7 100644
--- a/frontend/src/metabase/__support__/sample_dataset_fixture.js
+++ b/frontend/src/metabase/__support__/sample_dataset_fixture.js
@@ -1,6 +1,7 @@
 import Question from "metabase-lib/lib/Question";
 import { getMetadata } from "metabase/selectors/metadata";
 import { assocIn } from "icepick";
+import _ from "underscore";
 
 export const DATABASE_ID = 1;
 export const ANOTHER_DATABASE_ID = 2;
@@ -450,7 +451,7 @@ export const state = {
         values: []
       },
       '2': {
-        description: 'This is a unique ID for the product. It is also called the “Invoice number" or “Confirmation number" in customer facing emails and screens.',
+        description: 'This is a unique ID for the product. It is also called the “Invoice number” or “Confirmation number” in customer facing emails and screens.',
         table_id: 1,
         special_type: 'type/PK',
         name: 'ID',
@@ -1395,6 +1396,8 @@ export const state = {
 export const metadata = getMetadata(state);
 
 export const card = {
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "query",
         database: DATABASE_ID,
@@ -1405,6 +1408,8 @@ export const card = {
 };
 
 export const product_card = {
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "query",
         database: DATABASE_ID,
@@ -1417,6 +1422,8 @@ export const product_card = {
 export const orders_raw_card = {
     id: 1,
     name: "Raw orders data",
+    display: 'table',
+    visualization_settings: {},
     can_write: true,
     dataset_query: {
         type: "query",
@@ -1430,6 +1437,8 @@ export const orders_raw_card = {
 export const orders_count_card = {
     id: 2,
     name: "# orders data",
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "query",
         database: DATABASE_ID,
@@ -1443,6 +1452,8 @@ export const orders_count_card = {
 export const native_orders_count_card = {
     id: 2,
     name: "# orders data",
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "native",
         database: DATABASE_ID,
@@ -1455,6 +1466,8 @@ export const native_orders_count_card = {
 export const invalid_orders_count_card = {
     id: 2,
     name: "# orders data",
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "nosuchqueryprocessor",
         database: DATABASE_ID,
@@ -1468,6 +1481,8 @@ export const orders_count_by_id_card = {
     id: 2,
     name: "# orders data",
     can_write: false,
+    display: 'table',
+    visualization_settings: {},
     dataset_query: {
         type: "query",
         database: DATABASE_ID,
@@ -1525,6 +1540,37 @@ export function makeQuestion(fn = (card, state) => ({ card, state })) {
 }
 
 export const question = new Question(metadata, card);
+export const unsavedOrderCountQuestion = new Question(metadata, _.omit(orders_count_card, 'id'));
 export const productQuestion = new Question(metadata, product_card);
 const NoFieldsMetadata = getMetadata(assocIn(state, ["metadata", "tables", ORDERS_TABLE_ID, "fields"], []))
 export const questionNoFields = new Question(NoFieldsMetadata, card);
+
+export const orders_past_30_days_segment = {
+    "id": null,
+    "name": "Past 30 days",
+    "description": "Past 30 days created at",
+    "table_id": 1,
+    "definition": {
+        "source_table": 1,
+        "filter": ["time-interval", ["field-id", 1], -30, "day"]
+    }
+};
+
+export const vendor_count_metric = {
+    "id": null,
+    "name": "Vendor count",
+    "description": "Tells how many vendors we have",
+    "table_id": 3,
+    "definition": {
+        "aggregation": [
+            [
+                "distinct",
+                [
+                    "field-id",
+                    28
+                ]
+            ]
+        ],
+        "source_table": 3
+    }
+};
diff --git a/frontend/src/metabase/admin/databases/components/DeleteDatabaseModal.jsx b/frontend/src/metabase/admin/databases/components/DeleteDatabaseModal.jsx
index 1498d90312e97eaae8acec2407b2ca8b070dd528..908c31e197275cbd7759b849d58b8f3c37f669bf 100644
--- a/frontend/src/metabase/admin/databases/components/DeleteDatabaseModal.jsx
+++ b/frontend/src/metabase/admin/databases/components/DeleteDatabaseModal.jsx
@@ -23,6 +23,8 @@ export default class DeleteDatabaseModal extends Component {
     async deleteDatabase() {
         try {
             this.props.onDelete(this.props.database);
+            // immediately call on close because database deletion should be non blocking
+            this.props.onClose()
         } catch (error) {
             this.setState({ error });
         }
diff --git a/frontend/src/metabase/admin/databases/containers/DatabaseListApp.integ.spec.js b/frontend/src/metabase/admin/databases/containers/DatabaseListApp.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..9bff9910fcfc79f517f62d0408aa576bfd8122d4
--- /dev/null
+++ b/frontend/src/metabase/admin/databases/containers/DatabaseListApp.integ.spec.js
@@ -0,0 +1,70 @@
+import {
+    login,
+    createTestStore,
+} from "metabase/__support__/integrated_tests";
+
+import { mount } from "enzyme";
+import { FETCH_DATABASES, DELETE_DATABASE } from "metabase/admin/databases/database"
+import DatabaseListApp from "metabase/admin/databases/containers/DatabaseListApp";
+import { delay } from "metabase/lib/promise"
+
+import { MetabaseApi } from 'metabase/services'
+
+describe('dashboard list', () => {
+
+    beforeAll(async () => {
+        await login()
+    })
+
+    it('should render', async () => {
+        const store = await createTestStore()
+        store.pushPath("/admin/databases");
+
+        const app = mount(store.getAppContainer())
+
+        await store.waitForActions([FETCH_DATABASES])
+
+        const wrapper = app.find(DatabaseListApp)
+        expect(wrapper.length).toEqual(1)
+
+    })
+
+    describe('deletes', () => {
+        it('should not block deletes', async () => {
+            // mock the db_delete method call to simulate a longer running delete
+            MetabaseApi.db_delete = () => delay(5000)
+
+            const store = await createTestStore()
+            store.pushPath("/admin/databases");
+
+            const app = mount(store.getAppContainer())
+            await store.waitForActions([FETCH_DATABASES])
+
+            const wrapper = app.find(DatabaseListApp)
+            const dbCount = wrapper.find('tr').length
+
+            const deleteButton = wrapper.find('.Button.Button--danger').first()
+
+            deleteButton.simulate('click')
+
+            const deleteModal = wrapper.find('.test-modal')
+            deleteModal.find('.Form-input').simulate('change', { target: { value: "DELETE" }})
+            deleteModal.find('.Button.Button--danger').simulate('click')
+
+            // test that the modal is gone
+            expect(wrapper.find('.test-modal').length).toEqual(0)
+
+            // we should now have a disabled db row during delete
+            expect(wrapper.find('tr.disabled').length).toEqual(1)
+
+            // db delete finishes
+            await store.waitForActions([DELETE_DATABASE])
+
+            // there should be no disabled db rows now
+            expect(wrapper.find('tr.disabled').length).toEqual(0)
+
+            // we should now have one less database in the list
+            expect(wrapper.find('tr').length).toEqual(dbCount - 1)
+        })
+    })
+})
diff --git a/frontend/src/metabase/admin/databases/containers/DatabaseListApp.jsx b/frontend/src/metabase/admin/databases/containers/DatabaseListApp.jsx
index ab7241838a81395ecb91c0bcdb54cf1e90b276fd..c9ab23f2272ce899ad48c705bb6b801ea71d3f7b 100644
--- a/frontend/src/metabase/admin/databases/containers/DatabaseListApp.jsx
+++ b/frontend/src/metabase/admin/databases/containers/DatabaseListApp.jsx
@@ -23,7 +23,8 @@ const mapStateToProps = (state, props) => {
         created:              props.location.query.created,
         databases:            getDatabasesSorted(state),
         hasSampleDataset:     hasSampleDataset(state),
-        engines:              MetabaseSettings.get('engines')
+        engines:              MetabaseSettings.get('engines'),
+        deletes:              state.admin.databases.deletes
     }
 }
 
@@ -63,29 +64,41 @@ export default class DatabaseList extends Component {
                         </thead>
                         <tbody>
                             { databases ?
-                                databases.map(database =>
-                                    <tr key={database.id}>
-                                        <td>
-                                            <Link to={"/admin/databases/"+database.id} className="text-bold link">{database.name}</Link>
-                                        </td>
-                                        <td>
-                                            {engines && engines[database.engine] ? engines[database.engine]['driver-name'] : database.engine}
-                                        </td>
-                                        <td className="Table-actions">
-                                            <ModalWithTrigger
-                                                ref={"deleteDatabaseModal_"+database.id}
-                                                triggerClasses="Button Button--danger"
-                                                triggerElement="Delete"
-                                            >
-                                                <DeleteDatabaseModal
-                                                    database={database}
-                                                    onClose={() => this.refs["deleteDatabaseModal_"+database.id].close()}
-                                                    onDelete={() => this.props.deleteDatabase(database.id)}
-                                                />
-                                            </ModalWithTrigger>
-                                        </td>
-                                    </tr>
-                                )
+                                databases.map(database => {
+                                    const isDeleting = this.props.deletes.indexOf(database.id) !== -1
+                                    return (
+                                        <tr
+                                            key={database.id}
+                                            className={cx({'disabled': isDeleting })}
+                                        >
+                                            <td>
+                                                <Link to={"/admin/databases/"+database.id} className="text-bold link">
+                                                    {database.name}
+                                                </Link>
+                                            </td>
+                                            <td>
+                                                {engines && engines[database.engine] ? engines[database.engine]['driver-name'] : database.engine}
+                                            </td>
+                                            { isDeleting
+                                                ? (<td className="text-right">Deleting...</td>)
+                                                : (
+                                                    <td className="Table-actions">
+                                                        <ModalWithTrigger
+                                                            ref={"deleteDatabaseModal_"+database.id}
+                                                            triggerClasses="Button Button--danger"
+                                                            triggerElement="Delete"
+                                                        >
+                                                            <DeleteDatabaseModal
+                                                                database={database}
+                                                                onClose={() => this.refs["deleteDatabaseModal_"+database.id].close()}
+                                                                onDelete={() => this.props.deleteDatabase(database.id)}
+                                                            />
+                                                        </ModalWithTrigger>
+                                                    </td>
+                                                )
+                                            }
+                                        </tr>
+                                    )})
                             :
                                 <tr>
                                     <td colSpan={4}>
diff --git a/frontend/src/metabase/admin/databases/database.js b/frontend/src/metabase/admin/databases/database.js
index 62ee158324b822a483f2c65a47dce5a56e787d8a..84598f060221be8caf4df105798be651ea697ed2 100644
--- a/frontend/src/metabase/admin/databases/database.js
+++ b/frontend/src/metabase/admin/databases/database.js
@@ -11,11 +11,11 @@ import { MetabaseApi } from "metabase/services";
 
 const RESET = "metabase/admin/databases/RESET";
 const SELECT_ENGINE = "metabase/admin/databases/SELECT_ENGINE";
-const FETCH_DATABASES = "metabase/admin/databases/FETCH_DATABASES";
+export const FETCH_DATABASES = "metabase/admin/databases/FETCH_DATABASES";
 const INITIALIZE_DATABASE = "metabase/admin/databases/INITIALIZE_DATABASE";
 const ADD_SAMPLE_DATASET = "metabase/admin/databases/ADD_SAMPLE_DATASET";
 const SAVE_DATABASE = "metabase/admin/databases/SAVE_DATABASE";
-const DELETE_DATABASE = "metabase/admin/databases/DELETE_DATABASE";
+export const DELETE_DATABASE = "metabase/admin/databases/DELETE_DATABASE";
 const SYNC_DATABASE = "metabase/admin/databases/SYNC_DATABASE";
 
 export const reset = createAction(RESET);
@@ -110,15 +110,18 @@ export const saveDatabase = createThunkAction(SAVE_DATABASE, function(database,
     };
 });
 
+const START_DELETE = 'metabase/admin/databases/START_DELETE'
+const startDelete = createAction(START_DELETE)
+
+
 // deleteDatabase
-export const deleteDatabase = createThunkAction(DELETE_DATABASE, function(databaseId, redirect=false) {
+export const deleteDatabase = createThunkAction(DELETE_DATABASE, function(databaseId, redirect=true) {
     return async function(dispatch, getState) {
         try {
+            dispatch(startDelete(databaseId))
+            dispatch(push('/admin/databases/'));
             await MetabaseApi.db_delete({"dbId": databaseId});
             MetabaseAnalytics.trackEvent("Databases", "Delete", redirect ? "Using Detail" : "Using List");
-            if (redirect) {
-                dispatch(push('/admin/databases/'));
-            }
             return databaseId;
         } catch(error) {
             console.log('error deleting database', error);
@@ -156,6 +159,15 @@ const editingDatabase = handleActions({
     [SELECT_ENGINE]: { next: (state, { payload }) => ({...state, engine: payload }) }
 }, null);
 
+const deletes = handleActions({
+    [START_DELETE]: {
+        next: (state, { payload }) => state.concat([payload])
+    },
+    [DELETE_DATABASE]: {
+        next: (state, { payload }) => state.splice(state.indexOf(payload), 1)
+    }
+}, []);
+
 const DEFAULT_FORM_STATE = { formSuccess: null, formError: null };
 const formState = handleActions({
     [RESET]: { next: () => DEFAULT_FORM_STATE },
@@ -165,5 +177,6 @@ const formState = handleActions({
 export default combineReducers({
     databases,
     editingDatabase,
-    formState
+    formState,
+    deletes
 });
diff --git a/frontend/src/metabase/admin/people/components/GroupSelect.jsx b/frontend/src/metabase/admin/people/components/GroupSelect.jsx
index e92435ef8fdabb1962dd60072e1d38c5cdde7e36..60885db80c560b9ac28b322a846e07c3ae47caba 100644
--- a/frontend/src/metabase/admin/people/components/GroupSelect.jsx
+++ b/frontend/src/metabase/admin/people/components/GroupSelect.jsx
@@ -12,7 +12,10 @@ const GroupOption = ({ group, selectedGroups = {}, onGroupChange }) => {
     return (
         <div className={cx("GroupOption flex align-center p1 px2", { "cursor-pointer": !disabled })} onClick={() => !disabled && onGroupChange(group, !selected) }>
             <span className={cx("pr1", getGroupColor(group), { disabled })}>
-                <CheckBox checked={selected} borderColor="currentColor" size={18} />
+                <CheckBox
+                    checked={selected}
+                    size={18}
+                />
             </span>
             {group.name}
         </div>
diff --git a/frontend/src/metabase/admin/people/components/UserGroupSelect.jsx b/frontend/src/metabase/admin/people/components/UserGroupSelect.jsx
index b8e835c5b9426b60a90a541d2ae380c7b2063d86..40c8e115b4442c3c59e1e1bd5ae87b3cd6ea76c0 100644
--- a/frontend/src/metabase/admin/people/components/UserGroupSelect.jsx
+++ b/frontend/src/metabase/admin/people/components/UserGroupSelect.jsx
@@ -14,7 +14,10 @@ import GroupSummary from "./GroupSummary.jsx";
 const GroupOption = ({ name, color, selected, disabled, onChange }) =>
     <div className={cx("flex align-center p1 px2", { "cursor-pointer": !disabled })} onClick={() => !disabled && onChange(!selected) }>
         <span className={cx("pr1", color, { disabled })}>
-            <CheckBox checked={selected} borderColor="currentColor" size={18} />
+            <CheckBox
+                checked={selected}
+                size={18}
+            />
         </span>
         {name}
     </div>
diff --git a/frontend/src/metabase/admin/people/containers/PeopleListingApp.jsx b/frontend/src/metabase/admin/people/containers/PeopleListingApp.jsx
index 408f9c77a93b6fc7f8c1f0696d44813d9f1a327a..4d0a9af82e0b7026b8c1ade8d94ed592a74d0463 100644
--- a/frontend/src/metabase/admin/people/containers/PeopleListingApp.jsx
+++ b/frontend/src/metabase/admin/people/containers/PeopleListingApp.jsx
@@ -386,6 +386,10 @@ export default class PeopleListingApp extends Component {
                                         <Tooltip tooltip="Signed up via Google">
                                             <Icon name='google' />
                                         </Tooltip> : null}
+                                      {user.ldap_auth ?
+                                        <Tooltip tooltip="Signed up via LDAP">
+                                            <Icon name='ldap' />
+                                        </Tooltip> : null }
                                     </td>
                                     <td>{user.email}</td>
                                     <td>
diff --git a/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.integ.spec.js b/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..a89ee7d1823c94db14148394b383363142f899fa
--- /dev/null
+++ b/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.integ.spec.js
@@ -0,0 +1,48 @@
+import {
+    login,
+    createTestStore,
+    clickRouterLink,
+} from "metabase/__support__/integrated_tests";
+
+import { mount } from "enzyme";
+
+import SettingsEditorApp from "metabase/admin/settings/containers/SettingsEditorApp"
+import SettingsAuthenticationOptions from "metabase/admin/settings/components/SettingsAuthenticationOptions"
+import SettingsSingleSignOnForm from "../components/SettingsSingleSignOnForm.jsx";
+import SettingsLdapForm from "../components/SettingsLdapForm.jsx";
+
+import { INITIALIZE_SETTINGS } from "metabase/admin/settings/settings"
+
+describe('Admin Auth Options', () => {
+    beforeAll(async () => {
+        await login()
+    })
+
+    it('it should render the proper configuration form', async () => {
+        const store = await createTestStore()
+
+        store.pushPath("/admin/settings");
+
+        const app = mount(store.getAppContainer())
+        await store.waitForActions([INITIALIZE_SETTINGS])
+        const settingsWrapper = app.find(SettingsEditorApp)
+        const authListItem = settingsWrapper.find('span[children="Authentication"]')
+
+        clickRouterLink(authListItem)
+
+        expect(settingsWrapper.find(SettingsAuthenticationOptions).length).toBe(1)
+
+        // test google
+        const googleConfigButton = settingsWrapper.find('.Button').first()
+        clickRouterLink(googleConfigButton)
+
+        expect(settingsWrapper.find(SettingsSingleSignOnForm).length).toBe(1)
+
+        store.goBack()
+
+        // test ldap
+        const ldapConfigButton = settingsWrapper.find('.Button').last()
+        clickRouterLink(ldapConfigButton)
+        expect(settingsWrapper.find(SettingsLdapForm).length).toBe(1)
+    })
+})
diff --git a/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.jsx b/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..0a9e50735ed75670305daf44fb815db636927419
--- /dev/null
+++ b/frontend/src/metabase/admin/settings/components/SettingsAuthenticationOptions.jsx
@@ -0,0 +1,28 @@
+import React, { Component } from 'react'
+import { Link } from 'react-router'
+
+class SettingsAuthenticationOptions extends Component {
+    render () {
+        return (
+            <ul className="text-measure">
+                <li>
+                    <div className="bordered rounded shadowed bg-white p4">
+                        <h2>Sign in with Google</h2>
+                        <p>Allows users with existing Metabase accounts to login with a Google account that matches their email address in addition to their Metabase username and password.</p>
+                        <Link className="Button" to="/admin/settings/authentication/google">Configure</Link>
+                    </div>
+                </li>
+
+                <li className="mt2">
+                    <div className="bordered rounded shadowed bg-white p4">
+                        <h2>LDAP</h2>
+                        <p>Allows users within your LDAP directory to log in to Metabase with their LDAP credentials, and allows automatic mapping of LDAP groups to Metabase groups.</p>
+                        <Link className="Button" to="/admin/settings/authentication/ldap">Configure</Link>
+                    </div>
+                </li>
+            </ul>
+        )
+    }
+}
+
+export default SettingsAuthenticationOptions
diff --git a/frontend/src/metabase/admin/settings/components/SettingsLdapForm.jsx b/frontend/src/metabase/admin/settings/components/SettingsLdapForm.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..9a98e719c64a9be46b088a255e73668e7c1c1b8f
--- /dev/null
+++ b/frontend/src/metabase/admin/settings/components/SettingsLdapForm.jsx
@@ -0,0 +1,258 @@
+import React, { Component, PropTypes } from "react";
+
+import _ from "underscore";
+import cx from "classnames";
+
+import Collapse from "react-collapse";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs";
+import DisclosureTriangle from "metabase/components/DisclosureTriangle";
+import MetabaseUtils from "metabase/lib/utils";
+import SettingsSetting from "./SettingsSetting";
+
+export default class SettingsLdapForm extends Component {
+    constructor(props, context) {
+        super(props, context);
+        this.state = {
+            formData: {},
+            showAttributes: false,
+            submitting: "default",
+            valid: false,
+            validationErrors: {}
+        }
+    }
+
+    static propTypes = {
+        elements: PropTypes.array.isRequired,
+        formErrors: PropTypes.object,
+        updateLdapSettings: PropTypes.func.isRequired
+    };
+
+    componentWillMount() {
+        // this gives us an opportunity to load up our formData with any existing values for elements
+        this.updateFormData(this.props);
+    }
+
+    componentWillReceiveProps(nextProps) {
+        this.updateFormData(nextProps);
+    }
+
+    updateFormData(props) {
+        let formData = {};
+        for (const element of props.elements) {
+            formData[element.key] = element.value;
+        }
+        this.setState({ formData });
+    }
+
+    componentDidMount() {
+        this.validateForm();
+    }
+
+    componentDidUpdate() {
+        this.validateForm();
+    }
+
+    setSubmitting(submitting) {
+        this.setState({submitting});
+    }
+
+    setFormErrors(formErrors) {
+        this.setState({formErrors});
+    }
+
+    // return null if element passes validation, otherwise return an error message
+    validateElement([validationType, validationMessage], value, element) {
+        if (MetabaseUtils.isEmpty(value)) return;
+
+        switch (validationType) {
+            case "email":
+                return !MetabaseUtils.validEmail(value) ? (validationMessage || "That's not a valid email address") : null;
+            case "integer":
+                return isNaN(parseInt(value)) ? (validationMessage || "That's not a valid integer") : null;
+            case "ldap_filter":
+                return (value.match(/\(/g) || []).length !== (value.match(/\)/g) || []).length ? (validationMessage || "Check your parentheses") : null;
+        }
+    }
+
+    validateForm() {
+        let { elements } = this.props;
+        let { formData } = this.state;
+
+        let valid = true,
+            validationErrors = {};
+
+        // Validate form only if LDAP is enabled
+        if (formData['ldap-enabled']) {
+            elements.forEach(function(element) {
+                // test for required elements
+                if (element.required && MetabaseUtils.isEmpty(formData[element.key])) {
+                    valid = false;
+                }
+
+                if (element.validations) {
+                    element.validations.forEach(function(validation) {
+                        validationErrors[element.key] = this.validateElement(validation, formData[element.key], element);
+                        if (validationErrors[element.key]) valid = false;
+                    }, this);
+                }
+            }, this);
+        }
+
+        if (this.state.valid !== valid || !_.isEqual(this.state.validationErrors, validationErrors)) {
+            this.setState({ valid, validationErrors });
+        }
+    }
+
+    handleChangeEvent(element, value, event) {
+        this.setState((previousState) => ({ formData: {
+            ...previousState.formData,
+            [element.key]: (MetabaseUtils.isEmpty(value)) ? null : value
+        } }));
+    }
+
+    handleFormErrors(error) {
+        // parse and format
+        let formErrors = {};
+        if (error.data && error.data.message) {
+            formErrors.message = error.data.message;
+        } else {
+            formErrors.message = "Looks like we ran into some problems";
+        }
+
+        if (error.data && error.data.errors) {
+            formErrors.elements = error.data.errors;
+        }
+
+        return formErrors;
+    }
+
+    handleAttributeToggle() {
+        this.setState((previousState) => ({ showAttributes: !previousState['showAttributes'] }));
+    }
+
+    updateLdapSettings(e) {
+        e.preventDefault();
+
+        let { formData, valid } = this.state;
+
+        if (valid) {
+            this.setState({
+                formErrors: null,
+                submitting: "working"
+            });
+
+            this.props.updateLdapSettings(formData).then(() => {
+                this.setState({ submitting: "success" });
+
+                // show a confirmation for 3 seconds, then return to normal
+                setTimeout(() => this.setState({ submitting: "default" }), 3000);
+            }, (error) => {
+                this.setState({
+                    submitting: "default",
+                    formErrors: this.handleFormErrors(error)
+                });
+            });
+        }
+    }
+
+    render() {
+        const { elements } = this.props;
+        const { formData, formErrors, showAttributes, submitting, valid, validationErrors } = this.state;
+
+        let sections = {
+            'ldap-enabled': 'server',
+            'ldap-host': 'server',
+            'ldap-port': 'server',
+            'ldap-security': 'server',
+            'ldap-bind-dn': 'server',
+            'ldap-password': 'server',
+            'ldap-user-base': 'user',
+            'ldap-user-filter': 'user',
+            'ldap-attribute-email': 'attribute',
+            'ldap-attribute-firstname': 'attribute',
+            'ldap-attribute-lastname': 'attribute',
+            'ldap-group-sync': 'group',
+            'ldap-group-base': 'group'
+        }
+
+        const toElement = (element) => {
+            // merge together data from a couple places to provide a complete view of the Element state
+            let errorMessage = (formErrors && formErrors.elements) ? formErrors.elements[element.key] : validationErrors[element.key];
+            let value = formData[element.key] == null ? element.defaultValue : formData[element.key];
+
+            if (element.key === 'ldap-group-sync') {
+                return (
+                    <SettingsSetting
+                        key={element.key}
+                        setting={{ ...element, value }}
+                        updateSetting={this.handleChangeEvent.bind(this, element)}
+                        mappings={formData['ldap-group-mappings']}
+                        updateMappings={this.handleChangeEvent.bind(this, { key: 'ldap-group-mappings' })}
+                        errorMessage={errorMessage}
+                    />
+                );
+            }
+
+            return (
+                <SettingsSetting
+                    key={element.key}
+                    setting={{ ...element, value }}
+                    updateSetting={this.handleChangeEvent.bind(this, element)}
+                    errorMessage={errorMessage}
+                />
+            );
+        };
+
+        let serverSettings = elements.filter(e => sections[e.key] === 'server').map(toElement);
+        let userSettings = elements.filter(e => sections[e.key] === 'user').map(toElement);
+        let attributeSettings = elements.filter(e => sections[e.key] === 'attribute').map(toElement);
+        let groupSettings = elements.filter(e => sections[e.key] === 'group').map(toElement);
+
+        let saveSettingsButtonStates = {
+            default: "Save changes",
+            working: "Saving...",
+            success: "Changes saved!"
+        };
+
+        let disabled = (!valid || submitting !== "default");
+        let saveButtonText = saveSettingsButtonStates[submitting];
+
+        return (
+            <form noValidate>
+                <Breadcrumbs
+                    crumbs={[
+                        ["Authentication", "/admin/settings/authentication"],
+                        ["LDAP"]
+                    ]}
+                    className="ml2 mb3"
+                />
+                <h2 className="mx2">Server Settings</h2>
+                <ul>{serverSettings}</ul>
+                <h2 className="mx2">User Schema</h2>
+                <ul>{userSettings}</ul>
+                <div className="mb4">
+                    <div className="inline-block ml1 cursor-pointer text-brand-hover" onClick={this.handleAttributeToggle.bind(this)}>
+                        <div className="flex align-center">
+                            <DisclosureTriangle open={showAttributes} />
+                            <h3>Attributes</h3>
+                        </div>
+                    </div>
+                    <Collapse isOpened={showAttributes} keepCollapsedContent>
+                        <ul>{attributeSettings}</ul>
+                    </Collapse>
+                </div>
+                <h2 className="mx2">Group Schema</h2>
+                <ul>{groupSettings}</ul>
+                <div className="m2 mb4">
+                    <button className={cx("Button mr2", {"Button--primary": !disabled}, {"Button--success-new": submitting === "success"})} disabled={disabled} onClick={this.updateLdapSettings.bind(this)}>
+                        {saveButtonText}
+                    </button>
+                    { (formErrors && formErrors.message) ? (
+                        <span className="pl2 text-error text-bold">{formErrors.message}</span>
+                    ) : null }
+                </div>
+            </form>
+        );
+    }
+}
diff --git a/frontend/src/metabase/admin/settings/components/SettingsSingleSignOnForm.jsx b/frontend/src/metabase/admin/settings/components/SettingsSingleSignOnForm.jsx
index 722f5e81bc205c08ffaca4f30672101cae902c66..5203cf603db492d0db81e741f2ad708aeea6fa1b 100644
--- a/frontend/src/metabase/admin/settings/components/SettingsSingleSignOnForm.jsx
+++ b/frontend/src/metabase/admin/settings/components/SettingsSingleSignOnForm.jsx
@@ -4,6 +4,7 @@ import PropTypes from "prop-types";
 import cx from "classnames";
 import _ from "underscore";
 
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
 import Input from "metabase/components/Input.jsx";
 
 export default class SettingsSingleSignOnForm extends Component {
@@ -100,8 +101,17 @@ export default class SettingsSingleSignOnForm extends Component {
 
         return (
             <form noValidate>
-                <div className="px2"
-                     style={{maxWidth: "585px"}}>
+                <div
+                    className="px2"
+                    style={{maxWidth: "585px"}}
+                >
+                    <Breadcrumbs
+                        crumbs={[
+                            ["Authentication", "/admin/settings/authentication"],
+                            ["Google Sign-In"]
+                        ]}
+                        className="mb2"
+                    />
                     <h2>Sign in with Google</h2>
                     <p className="text-grey-4">
                         Allows users with existing Metabase accounts to login with a Google account that matches their email address in addition to their Metabase username and password.
diff --git a/frontend/src/metabase/admin/settings/components/widgets/LdapGroupMappingsWidget.jsx b/frontend/src/metabase/admin/settings/components/widgets/LdapGroupMappingsWidget.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..660cd988163f5024d59fe5e9816522a27bafc9f6
--- /dev/null
+++ b/frontend/src/metabase/admin/settings/components/widgets/LdapGroupMappingsWidget.jsx
@@ -0,0 +1,280 @@
+// @flow
+
+import React from "react";
+
+import { ModalFooter } from "metabase/components/ModalContent"
+import AdminContentTable from "metabase/components/AdminContentTable";
+import Button from "metabase/components/Button";
+import GroupSelect from "metabase/admin/people/components/GroupSelect";
+import GroupSummary from "metabase/admin/people/components/GroupSummary";
+import Icon from "metabase/components/Icon";
+import LoadingSpinner from "metabase/components/LoadingSpinner";
+import Modal from "metabase/components/Modal";
+import PopoverWithTrigger from "metabase/components/PopoverWithTrigger";
+
+import { PermissionsApi, SettingsApi } from "metabase/services";
+
+import _ from "underscore";
+
+import SettingToggle from './SettingToggle';
+
+type Props = {
+    setting: any,
+    updateSetting: (value: any) => void,
+    mappings: { [string]: number[] },
+    updateMappings: (value: { [string]: number[] }) => void
+};
+
+type State = {
+    showEditModal: boolean,
+    showAddRow: boolean,
+    groups: Object[],
+    mappings: { [string]: number[] }
+};
+
+export default class LdapGroupMappingsWidget extends React.Component {
+    props: Props;
+    state: State;
+
+    constructor(props: Props, context: any) {
+        super(props, context);
+        this.state = {
+            showEditModal: false,
+            showAddRow: false,
+            groups: [],
+            mappings: {}
+        };
+    }
+
+    _showEditModal = (e: Event) => {
+        e.preventDefault();
+        this.setState({ mappings: this.props.mappings || {}, showEditModal: true });
+        PermissionsApi.groups().then((groups) => this.setState({ groups }));
+    }
+
+    _showAddRow = (e: Event) => {
+        e.preventDefault();
+        this.setState({ showAddRow: true });
+    }
+
+    _hideAddRow = () => {
+        this.setState({ showAddRow: false });
+    }
+
+    _addMapping = (dn: string) => {
+        this.setState((prevState: State) => ({ mappings: { ...prevState.mappings, [dn]: [] }, showAddRow: false }));
+    }
+
+    _changeMapping = (dn: string) => (group: { id: number }, selected: boolean) => {
+        if (selected) {
+            this.setState((prevState: State) => ({
+                mappings: {
+                    ...prevState.mappings,
+                    [dn]: [...prevState.mappings[dn], group.id]
+                }
+            }));
+        } else {
+            this.setState((prevState: State) => ({
+                mappings: {
+                    ...prevState.mappings,
+                    [dn]: prevState.mappings[dn].filter(id => id !== group.id)
+                }
+            }));
+        }
+    }
+
+    _deleteMapping = (dn: string) => (e: Event) => {
+        e.preventDefault();
+        this.setState((prevState: State) => ({ mappings: _.omit(prevState.mappings, dn) }));
+    }
+
+    _cancelClick = (e: Event) => {
+        e.preventDefault();
+        this.setState({ showEditModal: false, showAddRow: false });
+    }
+
+    _saveClick = (e: Event) => {
+        e.preventDefault();
+        const { state: { mappings }, props: { updateMappings } } = this;
+        SettingsApi.put({ key: "ldap-group-mappings", value: mappings }).then(() => {
+            updateMappings && updateMappings(mappings);
+            this.setState({ showEditModal: false, showAddRow: false });
+        });
+    }
+
+    render() {
+        const { showEditModal, showAddRow, groups, mappings } = this.state;
+
+        return (
+            <div className="flex align-center">
+                <SettingToggle {...this.props} />
+                <div className="flex align-center pt1">
+                    <Button type="button" className="ml1" medium onClick={this._showEditModal}>Edit Mappings</Button>
+                </div>
+                { showEditModal ? (
+                    <Modal wide>
+                        <div>
+                            <div className="pt4 px4">
+                                <h2>Group Mappings</h2>
+                            </div>
+                            <div className="px4">
+                                <Button className="float-right" primary onClick={this._showAddRow}>Create a mapping</Button>
+                                <p className="text-measure">
+                                    Mappings allow Metabase to automatically add and remove users from groups based on the membership information provided by the
+                                    directory server. Membership to the Admin group can be granted through mappings, but will not be automatically removed as a
+                                    failsafe measure.
+                                </p>
+                                <AdminContentTable columnTitles={['Distinguished Name', 'Groups', '']}>
+                                    { showAddRow ? (
+                                        <AddMappingRow mappings={mappings} onCancel={this._hideAddRow} onAdd={this._addMapping} />
+                                    ) : null }
+                                    { ((Object.entries(mappings): any): Array<[string, number[]]>).map(([dn, ids]) =>
+                                        <MappingRow
+                                            key={dn}
+                                            dn={dn}
+                                            groups={groups}
+                                            selectedGroups={ids}
+                                            onChange={this._changeMapping(dn)}
+                                            onDelete={this._deleteMapping(dn)}
+                                        />
+                                    ) }
+                                </AdminContentTable>
+                            </div>
+                            <ModalFooter>
+                                <Button type="button" onClick={this._cancelClick}>Cancel</Button>
+                                <Button primary onClick={this._saveClick}>Save</Button>
+                            </ModalFooter>
+                        </div>
+                    </Modal>
+                ) : null }
+            </div>
+        );
+    }
+}
+
+type AddMappingRowProps = {
+    mappings: { [string]: number[] },
+    onAdd?: (dn: string) => void,
+    onCancel?: () => void
+};
+
+type AddMappingRowState = {
+    value: ''
+};
+
+class AddMappingRow extends React.Component {
+    props: AddMappingRowProps;
+    state: AddMappingRowState;
+
+    constructor(props: AddMappingRowProps, context: any) {
+        super(props, context);
+        this.state = {
+            value: ''
+        };
+    }
+
+    _handleCancelClick = (e: Event) => {
+        e.preventDefault();
+        const { onCancel } = this.props;
+        onCancel && onCancel();
+        this.setState({ value: '' });
+    }
+
+    _handleAddClick = (e: Event) => {
+        e.preventDefault();
+        const { onAdd } = this.props;
+        onAdd && onAdd(this.state.value);
+        this.setState({ value: '' });
+    }
+
+    render() {
+        const { value } = this.state;
+
+        const isValid = value && this.props.mappings[value] === undefined;
+
+        return (
+            <tr>
+                <td colSpan="3" style={{ padding: 0 }}>
+                    <div className="my2 pl1 p1 bordered border-brand rounded relative flex align-center">
+                        <input
+                            className="input--borderless h3 ml1 flex-full"
+                            type="text"
+                            value={value}
+                            placeholder="cn=People,ou=Groups,dc=metabase,dc=com"
+                            autoFocus
+                            onChange={(e) => this.setState({ value: e.target.value })}
+                        />
+                        <span className="link no-decoration cursor-pointer" onClick={this._handleCancelClick}>Cancel</span>
+                        <Button className="ml2" primary={!!isValid} disabled={!isValid} onClick={this._handleAddClick}>Add</Button>
+                    </div>
+                </td>
+            </tr>
+        );
+    }
+}
+
+class MappingGroupSelect extends React.Component {
+    props: {
+        groups: Array<{ id: number }>,
+        selectedGroups: number[],
+        onGroupChange?: (group: { id: number }, selected: boolean) => void
+    };
+
+    render() {
+        const { groups, selectedGroups, onGroupChange } = this.props;
+
+        if (!groups || groups.length === 0) {
+            return <LoadingSpinner />;
+        }
+
+        const selected = selectedGroups.reduce((g, id) => ({ ...g, [id]: true }), {});
+
+        return (
+            <PopoverWithTrigger
+                ref="popover"
+                triggerElement={
+                    <div className="flex align-center">
+                        <span className="mr1 text-grey-4">
+                            <GroupSummary groups={groups} selectedGroups={selected} />
+                        </span>
+                        <Icon className="text-grey-2" name="chevrondown"  size={10}/>
+                    </div>
+                }
+                triggerClasses="AdminSelectBorderless py1"
+                sizeToFit
+            >
+                <GroupSelect groups={groups} selectedGroups={selected} onGroupChange={onGroupChange} />
+            </PopoverWithTrigger>
+        );
+    }
+}
+
+class MappingRow extends React.Component {
+    props: {
+        dn: string,
+        groups: Array<{ id: number }>,
+        selectedGroups: number[],
+        onChange?: (group: { id: number }, selected: boolean) => void,
+        onDelete?: (e: Event) => void
+    };
+
+    render() {
+        const { dn, groups, selectedGroups, onChange, onDelete } = this.props;
+
+        return (
+            <tr>
+                <td>{dn}</td>
+                <td>
+                    <MappingGroupSelect
+                        groups={groups}
+                        selectedGroups={selectedGroups}
+                        onGroupChange={onChange}
+                    />
+                </td>
+                <td className="Table-actions">
+                    <Button warning onClick={onDelete}>Remove</Button>
+                </td>
+            </tr>
+        );
+    }
+}
diff --git a/frontend/src/metabase/admin/settings/containers/SettingsEditorApp.jsx b/frontend/src/metabase/admin/settings/containers/SettingsEditorApp.jsx
index e00c797bb5f1250185bc79cccdfdda4b67800b0e..1ca126d97073360adb479a6ba36f8f24f2e07e42 100644
--- a/frontend/src/metabase/admin/settings/containers/SettingsEditorApp.jsx
+++ b/frontend/src/metabase/admin/settings/containers/SettingsEditorApp.jsx
@@ -2,18 +2,20 @@ import React, { Component } from "react";
 import PropTypes from "prop-types";
 import { Link } from "react-router";
 import { connect } from "react-redux";
-
 import title from "metabase/hoc/Title";
 import MetabaseAnalytics from "metabase/lib/analytics";
+import { slugify } from "metabase/lib/formatting";
 
 import AdminLayout from "metabase/components/AdminLayout.jsx";
 
 import SettingsSetting from "../components/SettingsSetting.jsx";
 import SettingsEmailForm from "../components/SettingsEmailForm.jsx";
 import SettingsSlackForm from "../components/SettingsSlackForm.jsx";
+import SettingsLdapForm from "../components/SettingsLdapForm.jsx";
 import SettingsSetupList from "../components/SettingsSetupList.jsx";
 import SettingsUpdatesForm from "../components/SettingsUpdatesForm.jsx";
 import SettingsSingleSignOnForm from "../components/SettingsSingleSignOnForm.jsx";
+import SettingsAuthenticationOptions from "../components/SettingsAuthenticationOptions.jsx";
 
 import { prepareAnalyticsValue } from 'metabase/admin/settings/utils'
 
@@ -52,6 +54,7 @@ export default class SettingsEditorApp extends Component {
         updateSetting: PropTypes.func.isRequired,
         updateEmailSettings: PropTypes.func.isRequired,
         updateSlackSettings: PropTypes.func.isRequired,
+        updateLdapSettings: PropTypes.func.isRequired,
         sendTestEmail: PropTypes.func.isRequired
     };
 
@@ -138,15 +141,31 @@ export default class SettingsEditorApp extends Component {
                     updateSetting={this.updateSetting}
                 />
             );
-        } else if (activeSection.name === "Single Sign-On") {
-            return (
-                <SettingsSingleSignOnForm
-                    elements={activeSection.settings}
-                    updateSetting={this.updateSetting}
-                />
-            );
+        } else if (activeSection.name === "Authentication") {
+            // HACK - the presence of this param is a way for us to tell if
+            // a user is looking at a sub section of the autentication section
+            // since allowing for multi page settings more broadly would require
+            // a fairly significant refactor of how settings does its routing logic
+            if(this.props.params.authType) {
+                if(this.props.params.authType === 'ldap') {
+                    return (
+                        <SettingsLdapForm
+                            elements={_.findWhere(this.props.sections, { slug: 'ldap'}).settings}
+                            updateLdapSettings={this.props.updateLdapSettings}
+                        />
+                    )
+                } else if (this.props.params.authType === 'google') {
+                    return (
+                        <SettingsSingleSignOnForm
+                            elements={ _.findWhere(this.props.sections, { slug: slugify('Single Sign-On')}).settings}
+                            updateSetting={this.updateSetting}
+                        />
+                    )
+                }
+            } else {
+                return (<SettingsAuthenticationOptions />)
+            }
         } else {
-
             return (
                 <ul>
                     {activeSection.settings
@@ -172,6 +191,13 @@ export default class SettingsEditorApp extends Component {
         const { sections, activeSection, newVersionAvailable } = this.props;
 
         const renderedSections = _.map(sections, (section, idx) => {
+
+            // HACK - This is used to hide specific items in the sidebar and is currently
+            // only used as a way to fake the multi page auth settings pages without
+            // requiring a larger refactor.
+            if(section.sidebar === false) {
+                return false;
+            }
             const classes = cx("AdminList-item", "flex", "align-center", "justify-between", "no-decoration", {
                 "selected": activeSection && section.name === activeSection.name // this.state.currentSection === idx
             });
diff --git a/frontend/src/metabase/admin/settings/selectors.js b/frontend/src/metabase/admin/settings/selectors.js
index 0434679965a1356633557fbbc2c2da590b097b54..84c0957333d250f6961b4ed63804cfc888979d9b 100644
--- a/frontend/src/metabase/admin/settings/selectors.js
+++ b/frontend/src/metabase/admin/settings/selectors.js
@@ -12,6 +12,7 @@ import {
 } from "./components/widgets/PublicLinksListing.jsx";
 import SecretKeyWidget from "./components/widgets/SecretKeyWidget.jsx";
 import EmbeddingLegalese from "./components/widgets/EmbeddingLegalese";
+import LdapGroupMappingsWidget from "./components/widgets/LdapGroupMappingsWidget";
 
 import { UtilApi } from "metabase/services";
 
@@ -148,6 +149,7 @@ const SECTIONS = [
     },
     {
         name: "Single Sign-On",
+        sidebar: false,
         settings: [
             {
                 key: "google-auth-client-id"
@@ -157,6 +159,96 @@ const SECTIONS = [
             }
         ]
     },
+    {
+        name: "Authentication",
+        settings: []
+    },
+    {
+        name: "LDAP",
+        sidebar: false,
+        settings: [
+            {
+                key: "ldap-enabled",
+                display_name: "LDAP Authentication",
+                description: null,
+                type: "boolean"
+            },
+            {
+                key: "ldap-host",
+                display_name: "LDAP Host",
+                placeholder: "ldap.yourdomain.org",
+                type: "string",
+                required: true,
+                autoFocus: true
+            },
+            {
+                key: "ldap-port",
+                display_name: "LDAP Port",
+                placeholder: "389",
+                type: "string",
+                validations: [["integer", "That's not a valid port number"]]
+            },
+            {
+                key: "ldap-security",
+                display_name: "LDAP Security",
+                description: null,
+                type: "radio",
+                options: { none: "None", ssl: "SSL", starttls: "StartTLS" },
+                defaultValue: "none"
+            },
+            {
+                key: "ldap-bind-dn",
+                display_name: "Username or DN",
+                type: "string"
+            },
+            {
+                key: "ldap-password",
+                display_name: "Password",
+                type: "password"
+            },
+            {
+                key: "ldap-user-base",
+                display_name: "User search base",
+                type: "string",
+                required: true
+            },
+            {
+                key: "ldap-user-filter",
+                display_name: "User filter",
+                type: "string",
+                validations: [["ldap_filter", "Check your parentheses"]]
+            },
+            {
+                key: "ldap-attribute-email",
+                display_name: "Email attribute",
+                type: "string"
+            },
+            {
+                key: "ldap-attribute-firstname",
+                display_name: "First name attribute",
+                type: "string"
+            },
+            {
+                key: "ldap-attribute-lastname",
+                display_name: "Last name attribute",
+                type: "string"
+            },
+            {
+                key: "ldap-group-sync",
+                display_name: "Synchronize group memberships",
+                description: null,
+                widget: LdapGroupMappingsWidget
+            },
+            {
+                key: "ldap-group-base",
+                display_name: "Group search base",
+                type: "string"
+            },
+            {
+                key: "ldap-group-mappings"
+            }
+        ]
+    },
     {
         name: "Maps",
         settings: [
diff --git a/frontend/src/metabase/admin/settings/settings.js b/frontend/src/metabase/admin/settings/settings.js
index cc63d4829c5fc1a0545b1fe181c2c951900b4cc2..366c0879685f4e3e2b916171379803757587df8c 100644
--- a/frontend/src/metabase/admin/settings/settings.js
+++ b/frontend/src/metabase/admin/settings/settings.js
@@ -1,7 +1,7 @@
 
 import { createThunkAction, handleActions, combineReducers } from "metabase/lib/redux";
 
-import { SettingsApi, EmailApi, SlackApi } from "metabase/services";
+import { SettingsApi, EmailApi, SlackApi, LdapApi } from "metabase/services";
 
 import { refreshSiteSettings } from "metabase/redux/settings";
 
@@ -71,6 +71,19 @@ export const updateSlackSettings = createThunkAction(UPDATE_SLACK_SETTINGS, func
     };
 }, {});
 
+export const UPDATE_LDAP_SETTINGS = "metabase/admin/settings/UPDATE_LDAP_SETTINGS";
+export const updateLdapSettings = createThunkAction(UPDATE_LDAP_SETTINGS, function(settings) {
+    return async function(dispatch, getState) {
+        try {
+            await LdapApi.updateSettings(settings);
+            await dispatch(refreshSiteSettings());
+        } catch(error) {
+            console.log("error updating LDAP settings", settings, error);
+            throw error;
+        }
+    };
+});
+
 export const RELOAD_SETTINGS = "metabase/admin/settings/RELOAD_SETTINGS";
 export const reloadSettings = createThunkAction(RELOAD_SETTINGS, function() {
     return async function(dispatch, getState) {
diff --git a/frontend/src/metabase/auth/auth.js b/frontend/src/metabase/auth/auth.js
index b75b5db49a18de80cc2b5e3cf80e31f5d4ec263d..2e684474bf433ffb2642163c97b57e8c319adff2 100644
--- a/frontend/src/metabase/auth/auth.js
+++ b/frontend/src/metabase/auth/auth.js
@@ -6,6 +6,7 @@ import { push } from "react-router-redux";
 import MetabaseCookies from "metabase/lib/cookies";
 import MetabaseUtils from "metabase/lib/utils";
 import MetabaseAnalytics from "metabase/lib/analytics";
+import MetabaseSettings from "metabase/lib/settings";
 
 import { clearGoogleAuthCredentials } from "metabase/lib/auth";
 
@@ -19,7 +20,7 @@ export const LOGIN = "metabase/auth/LOGIN";
 export const login = createThunkAction(LOGIN, function(credentials, redirectUrl) {
     return async function(dispatch, getState) {
 
-        if (!MetabaseUtils.validEmail(credentials.email)) {
+        if (!MetabaseSettings.ldapEnabled() && !MetabaseUtils.validEmail(credentials.username)) {
             return {'data': {'errors': {'email': "Please enter a valid formatted email address."}}};
         }
 
diff --git a/frontend/src/metabase/auth/containers/LoginApp.jsx b/frontend/src/metabase/auth/containers/LoginApp.jsx
index 229febd771c0a21a04db540746c7fe5a041aeb16..a6c02e11903c620a4b85bb408f651459a839ff2b 100644
--- a/frontend/src/metabase/auth/containers/LoginApp.jsx
+++ b/frontend/src/metabase/auth/containers/LoginApp.jsx
@@ -11,7 +11,8 @@ import FormField from "metabase/components/form/FormField.jsx";
 import FormLabel from "metabase/components/form/FormLabel.jsx";
 import FormMessage from "metabase/components/form/FormMessage.jsx";
 import LogoIcon from "metabase/components/LogoIcon.jsx";
-import Settings from "metabase/lib/settings.js";
+import Settings from "metabase/lib/settings";
+import Utils from "metabase/lib/utils";
 
 
 import * as authActions from "../auth";
@@ -44,7 +45,7 @@ export default class LoginApp extends Component {
 
         let valid = true;
 
-        if (!credentials.email || !credentials.password) {
+        if (!credentials.username || !credentials.password) {
             valid = false;
         }
 
@@ -128,9 +129,9 @@ export default class LoginApp extends Component {
 
                             <FormMessage formError={loginError && loginError.data.message ? loginError : null} ></FormMessage>
 
-                            <FormField key="email" fieldName="email" formError={loginError}>
-                                <FormLabel title={"Email address"}  fieldName={"email"} formError={loginError} />
-                                <input className="Form-input Form-offset full py1" name="email" placeholder="youlooknicetoday@email.com" type="text" onChange={(e) => this.onChange("email", e.target.value)} autoFocus />
+                            <FormField key="username" fieldName="username" formError={loginError}>
+                                <FormLabel title={Settings.ldapEnabled() ? "Username or email address" : "Email address"} fieldName={"username"} formError={loginError} />
+                                <input className="Form-input Form-offset full py1" name="username" placeholder="youlooknicetoday@email.com" type="text" onChange={(e) => this.onChange("username", e.target.value)} autoFocus />
                                 <span className="Form-charm"></span>
                             </FormField>
 
@@ -150,7 +151,7 @@ export default class LoginApp extends Component {
                                 <button className={cx("Button Grid-cell", {'Button--primary': this.state.valid})} disabled={!this.state.valid}>
                                     Sign in
                                 </button>
-                                <Link to={"/auth/forgot_password"+(this.state.credentials.email ? "?email="+this.state.credentials.email : "")} className="Grid-cell py2 sm-py0 text-grey-3 md-text-right text-centered flex-full link" onClick={(e) => { window.OSX ? window.OSX.resetPassword() : null }}>I seem to have forgotten my password</Link>
+                                <Link to={"/auth/forgot_password"+(Utils.validEmail(this.state.credentials.username) ? "?email="+this.state.credentials.username : "")} className="Grid-cell py2 sm-py0 text-grey-3 md-text-right text-centered flex-full link" onClick={(e) => { window.OSX ? window.OSX.resetPassword() : null }}>I seem to have forgotten my password</Link>
                             </div>
                         </form>
                     </div>
diff --git a/frontend/src/metabase/components/AccordianList.jsx b/frontend/src/metabase/components/AccordianList.jsx
index e07a891c11a0a2eace9c23e0794ccf4bba21df25..bd3bc970c9726135e9972c08e2ba4b71e6f8d47c 100644
--- a/frontend/src/metabase/components/AccordianList.jsx
+++ b/frontend/src/metabase/components/AccordianList.jsx
@@ -65,7 +65,7 @@ export default class AccordianList extends Component {
     componentDidMount() {
         // when the component is mounted and an item is selected then scroll to it
         const element = this.refs.selected && ReactDOM.findDOMNode(this.refs.selected);
-        if (element && !elementIsInView(element)) {
+        if (element && !elementIsInView(element) && element.scrollIntoView) {
             element.scrollIntoView();
         }
     }
@@ -230,12 +230,11 @@ export default class AccordianList extends Component {
                                         className={cx("List-item flex", { 'List-item--selected': this.itemIsSelected(item, itemIndex), 'List-item--disabled': !this.itemIsClickable(item) }, this.getItemClasses(item, itemIndex))}
                                     >
                                         <a
-                                            className={cx("flex-full flex align-center px1", this.itemIsClickable(item) ? "cursor-pointer" : "cursor-default")}
-                                            style={{ paddingTop: "0.25rem", paddingBottom: "0.25rem" }}
+                                            className={cx("p1 flex-full flex align-center", this.itemIsClickable(item) ? "cursor-pointer" : "cursor-default")}
                                             onClick={this.itemIsClickable(item) && this.onChange.bind(this, item)}
                                         >
-                                            { this.renderItemIcon(item, itemIndex) }
-                                            <h4 className="List-item-title ml2">{item.name}</h4>
+                                            <span className="flex align-center">{ this.renderItemIcon(item, itemIndex) }</span>
+                                            <h4 className="List-item-title ml1">{item.name}</h4>
                                         </a>
                                         { this.renderItemExtra(item, itemIndex) }
                                         { showItemArrows &&
diff --git a/frontend/src/metabase/components/AdminAwareEmptyState.jsx b/frontend/src/metabase/components/AdminAwareEmptyState.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..6567dbcc5f5a1e6fa0059f6e252a7ebeffa341a2
--- /dev/null
+++ b/frontend/src/metabase/components/AdminAwareEmptyState.jsx
@@ -0,0 +1,32 @@
+import React from "react";
+import EmptyState from "metabase/components/EmptyState.jsx";
+/*
+ * AdminAwareEmptyState is a component that can
+ *  1) Produce a custom message for admins in empty results
+ */
+
+const AdminAwareEmptyState = ({user, title, message, adminMessage, icon, image, imageHeight, imageClassName, action, adminAction, link, adminLink, onActionClick, smallDescription = false}) =>
+         <EmptyState 
+            title={title}
+            message={user && user.is_superuser ?
+                adminMessage || message :
+                message
+            }
+            icon={icon}
+            image={image}
+            action={user && user.is_superuser ?
+                adminAction || action :
+                action
+            }
+            link={user && user.is_superuser ?
+                adminLink || link :
+                link
+            }
+            imageHeight={imageHeight}
+            imageClassName={imageClassName}
+            onActionClick={onActionClick}
+            smallDescription={smallDescription}
+        />
+
+
+export default AdminAwareEmptyState;
\ No newline at end of file
diff --git a/frontend/src/metabase/components/CheckBox.info.js b/frontend/src/metabase/components/CheckBox.info.js
index be9711d551258d1266898a28feaf373b1c35899f..aaec2b7213dac14c16024de4ca6a95c77ad8c37a 100644
--- a/frontend/src/metabase/components/CheckBox.info.js
+++ b/frontend/src/metabase/components/CheckBox.info.js
@@ -8,7 +8,10 @@ A standard checkbox.
 `;
 
 export const examples = {
-    "off": <CheckBox />,
-    "on": <CheckBox checked />,
-    "on inverted": <CheckBox style={{ color: "#509EE3" }} invertChecked checked />
+    "Default - Off": <CheckBox />,
+    "On - Default blue": <CheckBox checked />,
+    "Purple": <CheckBox checked color='purple' />,
+    "Yellow": <CheckBox checked color='yellow' />,
+    "Red": <CheckBox checked color='red' />,
+    "Green": <CheckBox checked color='green' />,
 };
diff --git a/frontend/src/metabase/components/CheckBox.jsx b/frontend/src/metabase/components/CheckBox.jsx
index d215052027bd764feb8d3a3ab2b567a258e71330..fc225a8c640a135807b2a4bb3c94c7d34b8ad107 100644
--- a/frontend/src/metabase/components/CheckBox.jsx
+++ b/frontend/src/metabase/components/CheckBox.jsx
@@ -1,20 +1,22 @@
 import React, { Component } from "react";
 import PropTypes from "prop-types";
-import Icon from "metabase/components/Icon.jsx";
+import Icon from "metabase/components/Icon";
 
-import cx from "classnames";
+import { normal as defaultColors } from "metabase/lib/colors";
 
 export default class CheckBox extends Component {
     static propTypes = {
         checked: PropTypes.bool,
-        onChange: PropTypes.func
+        onChange: PropTypes.func.isRequired,
+        color: PropTypes.oneOf(defaultColors),
+        size: PropTypes.number,  // TODO - this should probably be a concrete set of options
+        padding: PropTypes.number// TODO - the component should pad itself properly based on the size
     };
 
     static defaultProps = {
         size: 16,
         padding: 2,
-        borderColor: "#ddd",
-        checkColor: "currentColor"
+        color: 'blue'
     };
 
     onClick() {
@@ -25,21 +27,31 @@ export default class CheckBox extends Component {
     }
 
     render() {
-        const { checked, size, padding, borderColor, checkColor, className, invertChecked, style } = this.props;
+        const {
+            checked,
+            color,
+            padding,
+            size,
+        } = this.props;
+
+        const themeColor = defaultColors[color];
+
         const checkboxStyle = {
             width:              size,
             height:             size,
-            backgroundColor:    (invertChecked && checked) ? checkColor : "white",
-            border:             (invertChecked && checked) ? ("2px solid " + checkColor) : ("2px solid " + borderColor),
-            borderRadius:       4,
-            display:            "flex",
-            alignItems:         "center",
-            justifyContent:     "center",
+            backgroundColor:    checked ? themeColor : "white",
+            border:             `2px solid ${ checked ? themeColor : '#ddd' }`,
         };
         return (
-            <div style={style} className={cx("cursor-pointer", className)} onClick={() => this.onClick()}>
-                <div style={checkboxStyle}>
-                    { checked ? <Icon style={{ color: invertChecked ? "white" : checkColor }} name="check"  size={size - padding * 2} /> : null }
+            <div className="cursor-pointer" onClick={() => this.onClick()}>
+                <div style={checkboxStyle} className="flex align-center justify-center rounded">
+                    { checked && (
+                        <Icon
+                            style={{ color: checked ? 'white' : themeColor }}
+                            name="check"
+                            size={size - padding * 2}
+                        />
+                    )}
                 </div>
             </div>
         )
diff --git a/frontend/src/metabase/components/Logs.jsx b/frontend/src/metabase/components/Logs.jsx
index c6acd12eec6228c4d9c8e3961f1cc27bd2304ab8..5bf86b55f272465fd7e4b6cb105cf1686cd768bb 100644
--- a/frontend/src/metabase/components/Logs.jsx
+++ b/frontend/src/metabase/components/Logs.jsx
@@ -11,8 +11,8 @@ import "react-ansi-style/inject-css";
 import _ from "underscore";
 
 export default class Logs extends Component {
-    constructor(props, context) {
-        super(props, context);
+    constructor() {
+        super();
         this.state = {
             logs: [],
             scrollToBottom: true
@@ -31,12 +31,13 @@ export default class Logs extends Component {
         }, 500);
     }
 
+    async fetchLogs() {
+        let logs = await UtilApi.logs();
+        this.setState({ logs: logs.reverse() })
+    }
+
     componentWillMount() {
-        this.timer = setInterval(async () => {
-            let response = await UtilApi.logs();
-            let logs = await response.json()
-            this.setState({ logs: logs.reverse() })
-        }, 1000);
+        this.timer = setInterval(this.fetchLogs.bind(this), 1000);
     }
 
     componentDidMount() {
diff --git a/frontend/src/metabase/components/Logs.spec.js b/frontend/src/metabase/components/Logs.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..cba46350cf55808192a6dcff933ff177768b18c8
--- /dev/null
+++ b/frontend/src/metabase/components/Logs.spec.js
@@ -0,0 +1,29 @@
+import React from 'react'
+import Logs from './Logs'
+import { mount } from 'enzyme'
+import sinon from 'sinon'
+
+import { UtilApi } from 'metabase/services'
+
+describe('Logs', () => {
+    describe('log fetching', () => {
+        let timer
+
+        beforeEach(() => {
+            timer = sinon.useFakeTimers()
+        })
+
+        afterEach(() => {
+            timer.restore()
+        })
+
+        it('should call UtilApi.logs after 1 second', () => {
+            const wrapper = mount(<Logs />)
+            const utilSpy = sinon.spy(UtilApi, "logs")
+
+            expect(wrapper.state().logs.length).toEqual(0)
+            timer.tick(1001)
+            expect(utilSpy.called).toEqual(true)
+        })
+    })
+})
diff --git a/frontend/src/metabase/components/Modal.jsx b/frontend/src/metabase/components/Modal.jsx
index 662fd9fdc37916194890f71a3d10edc2e42f0761..7aec15576b845b3a8ae74a51494c9a215a72d2aa 100644
--- a/frontend/src/metabase/components/Modal.jsx
+++ b/frontend/src/metabase/components/Modal.jsx
@@ -172,6 +172,27 @@ export class InlineModal extends Component {
     }
 }
 
+/**
+ * A modified version of Modal for Jest/Enzyme tests. Renders the modal content inline instead of document root.
+ */
+export class TestModal extends Component {
+    render() {
+        if (this.props.isOpen) {
+            return (
+                <div className="test-modal">
+                    { getModalContent({
+                        ...this.props,
+                        fullPageModal: true,
+                        formModal: !!this.props.form
+                    }) }
+                </div>
+            )
+        } else {
+            return null;
+        }
+    }
+}
+
 // the "routeless" version should only be used for non-inline modals
 const RoutelessFullPageModal = routeless(FullPageModal);
 
diff --git a/frontend/src/metabase/components/Popover.jsx b/frontend/src/metabase/components/Popover.jsx
index cb306a0e0b89e3d8436b150211c9bfbea7795a0d..a850e156b451c489db2ac39e297348985bc83dd9 100644
--- a/frontend/src/metabase/components/Popover.jsx
+++ b/frontend/src/metabase/components/Popover.jsx
@@ -104,7 +104,7 @@ export default class Popover extends Component {
                 >
                     { typeof this.props.children === "function" ?
                         this.props.children()
-                    :
+                        :
                         this.props.children
                     }
                 </div>
@@ -274,3 +274,22 @@ export default class Popover extends Component {
         return <span className="hide" />;
     }
 }
+
+/**
+ * A modified version of TestPopover for Jest/Enzyme tests.
+ * Simply renders the popover body inline instead of mutating DOM root.
+ */
+export const TestPopover = (props) =>
+    props.isOpen ?
+        <div
+            id={props.id}
+            className={cx("TestPopover TestPopoverBody", props.className)}
+            style={props.style}
+        >
+            { typeof props.children === "function" ?
+                props.children()
+                :
+                props.children
+            }
+        </div>
+        : null
diff --git a/frontend/src/metabase/components/SidebarItem.jsx b/frontend/src/metabase/components/SidebarItem.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..adfbc982612e11ddbaa9d24e1a95b5c4f2657c22
--- /dev/null
+++ b/frontend/src/metabase/components/SidebarItem.jsx
@@ -0,0 +1,27 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import { Link } from "react-router";
+import S from "./Sidebar.css";
+
+import LabelIcon from "./LabelIcon.jsx";
+
+import pure from "recompose/pure";
+
+
+const SidebarItem = ({ name, sidebar, icon, href }) =>
+    <li>
+        <Link to={href} className={S.item} activeClassName={S.selected}>
+            <LabelIcon className={S.icon} icon={icon}/>
+            <span className={S.name}>{sidebar || name}</span>
+        </Link>
+    </li>
+
+SidebarItem.propTypes = {
+    name:  PropTypes.string.isRequired,
+    sidebar:  PropTypes.string,
+    icon:  PropTypes.string.isRequired,
+    href:  PropTypes.string.isRequired,
+};
+
+export default pure(SidebarItem);
diff --git a/frontend/src/metabase/components/StackedCheckBox.info.js b/frontend/src/metabase/components/StackedCheckBox.info.js
index 69a97375d36ae330d907dc25c0ed6f90a39fa0ca..433c096e213c6d7dd417381b7a5201ad7c2430a5 100644
--- a/frontend/src/metabase/components/StackedCheckBox.info.js
+++ b/frontend/src/metabase/components/StackedCheckBox.info.js
@@ -8,7 +8,7 @@ A stacked checkbox, representing "all" items.
 `;
 
 export const examples = {
-    "off": <StackedCheckBox />,
-    "on": <StackedCheckBox checked />,
-    "on inverted": <StackedCheckBox style={{ color: "#509EE3" }} invertChecked checked />
+    "Off - Default": <StackedCheckBox />,
+    "Checked": <StackedCheckBox checked />,
+    "Checked with color": <StackedCheckBox checked color='purple' />,
 };
diff --git a/frontend/src/metabase/components/StackedCheckBox.jsx b/frontend/src/metabase/components/StackedCheckBox.jsx
index 0332500ed3045a0edf069d2f80b68b3d51d1fd82..1567dfd9e8ddc2504f97ad709083f78009b50336 100644
--- a/frontend/src/metabase/components/StackedCheckBox.jsx
+++ b/frontend/src/metabase/components/StackedCheckBox.jsx
@@ -1,11 +1,21 @@
 import React from "react";
-
 import CheckBox from "metabase/components/CheckBox.jsx";
 
+const OFFSET = 4;
+
 const StackedCheckBox = (props) =>
-    <span className={props.className} style={{ ...props.style, position: "relative" }}>
-        <CheckBox {...props} className={null} style={{ position: "absolute", top: -3, left: 3, zIndex: -1 }} />
-        <CheckBox {...props} className={null} style={{}} />
-    </span>
+    <div className="relative">
+        <span
+            className="absolute"
+            style={{
+                top: -OFFSET,
+                left: OFFSET,
+                zIndex: -1
+            }}
+        >
+            <CheckBox {...props} />
+        </span>
+        <CheckBox {...props} />
+    </div>
 
 export default StackedCheckBox;
diff --git a/frontend/src/metabase/components/Tooltip.jsx b/frontend/src/metabase/components/Tooltip.jsx
index 7b8d6f5fbb4c1adf7ba7f639195a044462f45704..86c0295a78675c0c4ee1e137c04e49312361b260 100644
--- a/frontend/src/metabase/components/Tooltip.jsx
+++ b/frontend/src/metabase/components/Tooltip.jsx
@@ -82,11 +82,79 @@ export default class Tooltip extends Component {
     _onMouseUp = (e) => {
         // This is in a timeout to ensure the component has a chance to fully unmount
         this.timer = setTimeout(() =>
-            this.setState({ isOpen: this.state.isHovered })
-        , 0);
+                this.setState({ isOpen: this.state.isHovered })
+            , 0);
     }
 
     render() {
         return React.Children.only(this.props.children);
     }
 }
+
+/**
+ * Modified version of Tooltip for Jest/Enzyme tests. Instead of manipulating the document root it
+ * renders the tooltip content (in TestTooltipContent) next to "children" / hover area (TestTooltipHoverArea).
+ *
+ * The test tooltip can only be toggled with `jestWrapper.simulate("mouseenter")` and `jestWrapper.simulate("mouseleave")`.
+ */
+export class TestTooltip extends Component {
+    constructor(props, context) {
+        super(props, context);
+
+        this.state = {
+            isOpen: false,
+            isHovered: false
+        };
+    }
+
+    static propTypes = {
+        tooltip: PropTypes.node,
+        children: PropTypes.element.isRequired,
+        isEnabled: PropTypes.bool,
+        verticalAttachments: PropTypes.array,
+        isOpen: PropTypes.bool
+    };
+
+    static defaultProps = {
+        isEnabled: true,
+        verticalAttachments: ["top", "bottom"]
+    };
+
+    _onMouseEnter = (e) => {
+        this.setState({ isOpen: true, isHovered: true });
+    }
+
+    _onMouseLeave = (e) => {
+        this.setState({ isOpen: false, isHovered: false });
+    }
+
+    render() {
+        const { isEnabled, tooltip } = this.props;
+        const isOpen = this.props.isOpen != null ? this.props.isOpen : this.state.isOpen;
+
+        return (
+            <div>
+                <TestTooltipTarget
+                    onMouseEnter={this._onMouseEnter}
+                    onMouseLeave={this._onMouseLeave}
+                >
+                    {this.props.children}
+                </TestTooltipTarget>
+
+                { tooltip && isEnabled && isOpen &&
+                    <TestTooltipContent>
+                        <TooltipPopover isOpen={true} target={this} {...this.props} children={this.props.tooltip} />
+                        {this.props.tooltip}
+                    </TestTooltipContent>
+                }
+            </div>
+        )
+    }
+}
+
+export const TestTooltipTarget = ({ children, onMouseEnter, onMouseLeave }) =>
+    <div className="test-tooltip-hover-area" onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave}>
+        {children}
+    </div>
+
+export const TestTooltipContent = ({ children }) => <div className="test-tooltip-content">{children}</div>
diff --git a/frontend/src/metabase/css/components/list.css b/frontend/src/metabase/css/components/list.css
index be2a8a8b70b2ac67a1adfe2cfeae2ec351f89d1f..21b6f3d7b9d0b9416e1acbabc4f5b86814f96b6f 100644
--- a/frontend/src/metabase/css/components/list.css
+++ b/frontend/src/metabase/css/components/list.css
@@ -37,8 +37,7 @@
 
 .List-item {
     display: flex;
-    border-radius: 6px;
-    border: 2px solid transparent;
+    border-radius: 4px;
     margin-top: 2px;
     margin-bottom: 2px;
 }
diff --git a/frontend/src/metabase/css/core/base.css b/frontend/src/metabase/css/core/base.css
index 6ed9b2935a3893690157e4e8a137c034fb17492f..bc19008d1e96161ddebe4eb394d30987483a7059 100644
--- a/frontend/src/metabase/css/core/base.css
+++ b/frontend/src/metabase/css/core/base.css
@@ -17,8 +17,10 @@ body {
     height: 100%; /* ensure the entire page will fill the window */
     display: flex;
     flex-direction: column;
+
     text-rendering: optimizeLegibility;
     -webkit-font-smoothing: antialiased;
+    -moz-osx-font-smoothing: grayscale;
 }
 
 /*
diff --git a/frontend/src/metabase/css/core/hide.css b/frontend/src/metabase/css/core/hide.css
index c3daca5b9b3a6d9d35621ad2bf917602514e158b..58ac6c1cdc5073c497ca5c53e55113b86fd78ace 100644
--- a/frontend/src/metabase/css/core/hide.css
+++ b/frontend/src/metabase/css/core/hide.css
@@ -1,5 +1,5 @@
 .hide { display: none !important; }
-.show { display: inheirt; }
+.show { display: inherit; }
 
 .hidden { visibility: hidden; }
 
diff --git a/frontend/src/metabase/css/query_builder.css b/frontend/src/metabase/css/query_builder.css
index e2cdcdbe8c773e0328f2f2d3eceed2569cc84988..7162c0e97cfb780683ab9775b41785436fdd00a5 100644
--- a/frontend/src/metabase/css/query_builder.css
+++ b/frontend/src/metabase/css/query_builder.css
@@ -243,25 +243,25 @@
 .QueryError-image--noRows {
   width: 120px;
   height: 120px;
-  background-image: url('../assets/img/no_results.svg');
+  background-image: url('../app/assets/img/no_results.svg');
 }
 
 .QueryError-image--queryError {
   width: 120px;
   height: 120px;
-  background-image: url('../assets/img/no_understand.svg');
+  background-image: url('../app/assets/img/no_understand.svg');
 }
 
 .QueryError-image--serverError {
   width: 120px;
   height: 148px;
-  background-image: url('../assets/img/blown_up.svg');
+  background-image: url('../app/assets/img/blown_up.svg');
 }
 
 .QueryError-image--timeout {
   width: 120px;
   height: 120px;
-  background-image: url('../assets/img/stopwatch.svg');
+  background-image: url('../app/assets/img/stopwatch.svg');
 }
 
 .QueryError-message {
diff --git a/frontend/src/metabase/dashboard/containers/DashCardCardParameterMapper.jsx b/frontend/src/metabase/dashboard/containers/DashCardCardParameterMapper.jsx
index b6d7354131b79aed67036368801438b9b024362f..3e4312213afaa1cfaab6fc58cc88d00d733f01bf 100644
--- a/frontend/src/metabase/dashboard/containers/DashCardCardParameterMapper.jsx
+++ b/frontend/src/metabase/dashboard/containers/DashCardCardParameterMapper.jsx
@@ -69,6 +69,7 @@ export default class DashCardCardParameterMapper extends Component {
     componentDidMount() {
         const { card } = this.props;
         // Type check for Flow
+
         card.dataset_query instanceof AtomicQuery && this.props.fetchDatabaseMetadata(card.dataset_query.database);
     }
 
diff --git a/frontend/src/metabase/home/containers/HomepageApp.integ.spec.js b/frontend/src/metabase/home/containers/HomepageApp.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..4af645adef4f2564a1edda2bd7cb3b4b0d5d88fd
--- /dev/null
+++ b/frontend/src/metabase/home/containers/HomepageApp.integ.spec.js
@@ -0,0 +1,86 @@
+import {
+    login,
+    createTestStore, createSavedQuestion, clickRouterLink
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from "enzyme";
+import {
+    orders_past_30_days_segment,
+    unsavedOrderCountQuestion,
+    vendor_count_metric
+} from "metabase/__support__/sample_dataset_fixture";
+import { delay } from 'metabase/lib/promise';
+
+import HomepageApp from "metabase/home/containers/HomepageApp";
+import { createMetric, createSegment } from "metabase/admin/datamodel/datamodel";
+import { FETCH_ACTIVITY } from "metabase/home/actions";
+import { QUERY_COMPLETED } from "metabase/query_builder/actions";
+
+import Activity from "metabase/home/components/Activity";
+import ActivityItem from "metabase/home/components/ActivityItem";
+import ActivityStory from "metabase/home/components/ActivityStory";
+import Scalar from "metabase/visualizations/visualizations/Scalar";
+
+describe("HomepageApp", () => {
+    beforeAll(async () => {
+        await login()
+
+        // Create some entities that will show up in the top of activity feed
+        // This test doesn't care if there already are existing items in the feed or not
+        // Delays are required for having separable creation times for each entity
+        await createSavedQuestion(unsavedOrderCountQuestion)
+        await delay(100);
+        await createSegment(orders_past_30_days_segment);
+        await delay(100);
+        await createMetric(vendor_count_metric);
+        await delay(100);
+    })
+
+    describe("activity feed", async () => {
+        it("shows the expected list of activity", async () => {
+            const store = await createTestStore()
+
+            store.pushPath("/");
+            const homepageApp = mount(store.connectContainer(<HomepageApp />));
+            await store.waitForActions([FETCH_ACTIVITY])
+
+            const activityFeed = homepageApp.find(Activity);
+            const activityItems = activityFeed.find(ActivityItem);
+            const activityStories = activityFeed.find(ActivityStory);
+
+            expect(activityItems.length).toBeGreaterThanOrEqual(3);
+            expect(activityStories.length).toBeGreaterThanOrEqual(3);
+
+            expect(activityItems.at(0).text()).toMatch(/Vendor count/);
+            expect(activityStories.at(0).text()).toMatch(/Tells how many vendors we have/);
+
+            expect(activityItems.at(1).text()).toMatch(/Past 30 days/);
+            expect(activityStories.at(1).text()).toMatch(/Past 30 days created at/);
+
+            // eslint-disable-next-line no-irregular-whitespace
+            expect(activityItems.at(2).text()).toMatch(/You saved a question about Orders/);
+            expect(activityStories.at(2).text()).toMatch(new RegExp(unsavedOrderCountQuestion.displayName()));
+
+
+        });
+        it("shows successfully open QB for a metric when clicking the metric name", async () => {
+            const store = await createTestStore()
+
+            store.pushPath("/");
+
+            // In this test we have to render the whole app in order to get links work properly
+            const app = mount(store.getAppContainer())
+            await store.waitForActions([FETCH_ACTIVITY])
+            const homepageApp = app.find(HomepageApp);
+
+            const activityFeed = homepageApp.find(Activity);
+            const metricLink = activityFeed.find(ActivityItem).find('a[children="Vendor count"]').first();
+            clickRouterLink(metricLink)
+            
+            await store.waitForActions([QUERY_COMPLETED]);
+            expect(app.find(Scalar).text()).toBe("200");
+        })
+    });
+
+});
diff --git a/frontend/src/metabase/icon_paths.js b/frontend/src/metabase/icon_paths.js
index 1d090d3c0340d3360a15f4d847738337cd514761..9c658befe881179fca8f13b1917ae7c912cafcaa 100644
--- a/frontend/src/metabase/icon_paths.js
+++ b/frontend/src/metabase/icon_paths.js
@@ -126,6 +126,10 @@ export var ICON_PATHS = {
         attrs: { fillRule: "evenodd" }
     },
     label: 'M14.577 31.042a2.005 2.005 0 0 1-2.738-.733L1.707 12.759c-.277-.477-.298-1.265-.049-1.757L6.45 1.537C6.7 1.044 7.35.67 7.9.7l10.593.582c.551.03 1.22.44 1.498.921l10.132 17.55a2.002 2.002 0 0 1-.734 2.737l-14.812 8.552zm.215-22.763a3.016 3.016 0 1 0-5.224 3.016 3.016 3.016 0 0 0 5.224-3.016z',
+    ldap: {
+        path: 'M1.006 3h13.702c.554 0 1.178.41 1.39.915l.363.874c.21.504.827.915 1.376.915h13.169c.54 0 .994.448.994 1.001v20.952a.99.99 0 0 1-.992 1H1.002c-.54 0-.993-.45-.993-1.005l-.01-23.646C0 3.446.45 3 1.007 3zM16.5 19.164c1.944 0 3.52-1.828 3.52-4.082 0-2.254-1.576-4.082-3.52-4.082-1.945 0-3.52 1.828-3.52 4.082 0 2.254 1.575 4.082 3.52 4.082zm6.5 4.665c0-1.872-1.157-3.521-2.913-4.484-.927.97-2.192 1.568-3.587 1.568s-2.66-.597-3.587-1.568C11.157 20.308 10 21.957 10 23.83h13z',
+        attrs: { fillRule: 'evenodd' }
+    },
     left: "M21,0 L5,16 L21,32 L21,5.47117907e-13 L21,0 Z",
     link: "M12.56 17.04c-1.08 1.384-1.303 1.963 1.755 4.04 3.058 2.076 7.29.143 8.587-1.062 1.404-1.304 4.81-4.697 7.567-7.842 2.758-3.144 1.338-8.238-.715-9.987-5.531-4.71-9.5-.554-11.088.773-2.606 2.176-5.207 5.144-5.207 5.144s1.747-.36 2.784 0c1.036.36 2.102.926 2.102.926l4.003-3.969s2.367-1.907 4.575 0 .674 4.404 0 5.189c-.674.784-6.668 6.742-6.668 6.742s-1.52.811-2.37.811c-.85 0-2.582-.528-2.582-.932 0-.405-1.665-1.22-2.744.166zm7.88-2.08c1.08-1.384 1.303-1.963-1.755-4.04-3.058-2.076-7.29-.143-8.587 1.062-1.404 1.304-4.81 4.697-7.567 7.842-2.758 3.144-1.338 8.238.715 9.987 5.531 4.71 9.5.554 11.088-.773 2.606-2.176 5.207-5.144 5.207-5.144s-1.747.36-2.784 0a17.379 17.379 0 0 1-2.102-.926l-4.003 3.969s-2.367 1.907-4.575 0-.674-4.404 0-5.189c.674-.784 6.668-6.742 6.668-6.742s1.52-.811 2.37-.811c.85 0 2.582.528 2.582.932 0 .405 1.665 1.22 2.744-.166z",
     line: 'M18.867 16.377l-3.074-3.184-.08.077-.002-.002.01-.01-.53-.528-.066-.07-.001.002-2.071-2.072L-.002 23.645l2.668 2.668 10.377-10.377 3.074 3.183.08-.076.001.003-.008.008.5.501.094.097.002-.001 2.072 2.072L31.912 8.669 29.244 6 18.867 16.377z',
diff --git a/frontend/src/metabase/internal/__snapshots__/components.spec.js.snap b/frontend/src/metabase/internal/__snapshots__/components.spec.js.snap
index ecd542cc21ab2618366225fc7e38c539440603d6..a815f23cff41b4a3cdb6f87754c79987796a649e 100644
--- a/frontend/src/metabase/internal/__snapshots__/components.spec.js.snap
+++ b/frontend/src/metabase/internal/__snapshots__/components.spec.js.snap
@@ -55,22 +55,18 @@ exports[`Button should render "with an icon" correctly 1`] = `
 </button>
 `;
 
-exports[`CheckBox should render "off" correctly 1`] = `
+exports[`CheckBox should render "Default - Off" correctly 1`] = `
 <div
   className="cursor-pointer"
   onClick={[Function]}
-  style={undefined}
 >
   <div
+    className="flex align-center justify-center rounded"
     style={
       Object {
-        "alignItems": "center",
         "backgroundColor": "white",
         "border": "2px solid #ddd",
-        "borderRadius": 4,
-        "display": "flex",
         "height": 16,
-        "justifyContent": "center",
         "width": 16,
       }
     }
@@ -78,26 +74,18 @@ exports[`CheckBox should render "off" correctly 1`] = `
 </div>
 `;
 
-exports[`CheckBox should render "on inverted" correctly 1`] = `
+exports[`CheckBox should render "Green" correctly 1`] = `
 <div
   className="cursor-pointer"
   onClick={[Function]}
-  style={
-    Object {
-      "color": "#509EE3",
-    }
-  }
 >
   <div
+    className="flex align-center justify-center rounded"
     style={
       Object {
-        "alignItems": "center",
-        "backgroundColor": "currentColor",
-        "border": "2px solid currentColor",
-        "borderRadius": 4,
-        "display": "flex",
+        "backgroundColor": "#9CC177",
+        "border": "2px solid #9CC177",
         "height": 16,
-        "justifyContent": "center",
         "width": 16,
       }
     }
@@ -124,22 +112,18 @@ exports[`CheckBox should render "on inverted" correctly 1`] = `
 </div>
 `;
 
-exports[`CheckBox should render "on" correctly 1`] = `
+exports[`CheckBox should render "On - Default blue" correctly 1`] = `
 <div
   className="cursor-pointer"
   onClick={[Function]}
-  style={undefined}
 >
   <div
+    className="flex align-center justify-center rounded"
     style={
       Object {
-        "alignItems": "center",
-        "backgroundColor": "white",
-        "border": "2px solid #ddd",
-        "borderRadius": 4,
-        "display": "flex",
+        "backgroundColor": "#509EE3",
+        "border": "2px solid #509EE3",
         "height": 16,
-        "justifyContent": "center",
         "width": 16,
       }
     }
@@ -152,7 +136,7 @@ exports[`CheckBox should render "on" correctly 1`] = `
       size={12}
       style={
         Object {
-          "color": "currentColor",
+          "color": "white",
         }
       }
       viewBox="0 0 32 32"
@@ -166,136 +150,181 @@ exports[`CheckBox should render "on" correctly 1`] = `
 </div>
 `;
 
-exports[`StackedCheckBox should render "off" correctly 1`] = `
-<span
-  className={undefined}
-  style={
-    Object {
-      "position": "relative",
-    }
-  }
+exports[`CheckBox should render "Purple" correctly 1`] = `
+<div
+  className="cursor-pointer"
+  onClick={[Function]}
 >
   <div
-    className="cursor-pointer"
-    onClick={[Function]}
+    className="flex align-center justify-center rounded"
     style={
       Object {
-        "left": 3,
-        "position": "absolute",
-        "top": -3,
-        "zIndex": -1,
+        "backgroundColor": "#A989C5",
+        "border": "2px solid #A989C5",
+        "height": 16,
+        "width": 16,
       }
     }
   >
-    <div
+    <svg
+      className="Icon Icon-check"
+      fill="currentcolor"
+      height={12}
+      name="check"
+      size={12}
       style={
         Object {
-          "alignItems": "center",
-          "backgroundColor": "white",
-          "border": "2px solid #ddd",
-          "borderRadius": 4,
-          "display": "flex",
-          "height": 16,
-          "justifyContent": "center",
-          "width": 16,
+          "color": "white",
         }
       }
-    />
+      viewBox="0 0 32 32"
+      width={12}
+    >
+      <path
+        d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
+      />
+    </svg>
   </div>
+</div>
+`;
+
+exports[`CheckBox should render "Red" correctly 1`] = `
+<div
+  className="cursor-pointer"
+  onClick={[Function]}
+>
   <div
-    className="cursor-pointer"
-    onClick={[Function]}
-    style={Object {}}
+    className="flex align-center justify-center rounded"
+    style={
+      Object {
+        "backgroundColor": "#EF8C8C",
+        "border": "2px solid #EF8C8C",
+        "height": 16,
+        "width": 16,
+      }
+    }
   >
-    <div
+    <svg
+      className="Icon Icon-check"
+      fill="currentcolor"
+      height={12}
+      name="check"
+      size={12}
       style={
         Object {
-          "alignItems": "center",
-          "backgroundColor": "white",
-          "border": "2px solid #ddd",
-          "borderRadius": 4,
-          "display": "flex",
-          "height": 16,
-          "justifyContent": "center",
-          "width": 16,
+          "color": "white",
         }
       }
-    />
+      viewBox="0 0 32 32"
+      width={12}
+    >
+      <path
+        d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
+      />
+    </svg>
   </div>
-</span>
+</div>
 `;
 
-exports[`StackedCheckBox should render "on inverted" correctly 1`] = `
-<span
-  className={undefined}
-  style={
-    Object {
-      "color": "#509EE3",
-      "position": "relative",
-    }
-  }
+exports[`CheckBox should render "Yellow" correctly 1`] = `
+<div
+  className="cursor-pointer"
+  onClick={[Function]}
 >
   <div
-    className="cursor-pointer"
-    onClick={[Function]}
+    className="flex align-center justify-center rounded"
     style={
       Object {
-        "left": 3,
-        "position": "absolute",
-        "top": -3,
-        "zIndex": -1,
+        "backgroundColor": "#f9d45c",
+        "border": "2px solid #f9d45c",
+        "height": 16,
+        "width": 16,
       }
     }
   >
-    <div
+    <svg
+      className="Icon Icon-check"
+      fill="currentcolor"
+      height={12}
+      name="check"
+      size={12}
       style={
         Object {
-          "alignItems": "center",
-          "backgroundColor": "currentColor",
-          "border": "2px solid currentColor",
-          "borderRadius": 4,
-          "display": "flex",
-          "height": 16,
-          "justifyContent": "center",
-          "width": 16,
+          "color": "white",
         }
       }
+      viewBox="0 0 32 32"
+      width={12}
     >
-      <svg
-        className="Icon Icon-check"
-        fill="currentcolor"
-        height={12}
-        name="check"
-        size={12}
+      <path
+        d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
+      />
+    </svg>
+  </div>
+</div>
+`;
+
+exports[`StackedCheckBox should render "Checked with color" correctly 1`] = `
+<div
+  className="relative"
+>
+  <span
+    className="absolute"
+    style={
+      Object {
+        "left": 4,
+        "top": -4,
+        "zIndex": -1,
+      }
+    }
+  >
+    <div
+      className="cursor-pointer"
+      onClick={[Function]}
+    >
+      <div
+        className="flex align-center justify-center rounded"
         style={
           Object {
-            "color": "white",
+            "backgroundColor": "#A989C5",
+            "border": "2px solid #A989C5",
+            "height": 16,
+            "width": 16,
           }
         }
-        viewBox="0 0 32 32"
-        width={12}
       >
-        <path
-          d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
-        />
-      </svg>
+        <svg
+          className="Icon Icon-check"
+          fill="currentcolor"
+          height={12}
+          name="check"
+          size={12}
+          style={
+            Object {
+              "color": "white",
+            }
+          }
+          viewBox="0 0 32 32"
+          width={12}
+        >
+          <path
+            d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
+          />
+        </svg>
+      </div>
     </div>
-  </div>
+  </span>
   <div
     className="cursor-pointer"
     onClick={[Function]}
-    style={Object {}}
   >
     <div
+      className="flex align-center justify-center rounded"
       style={
         Object {
-          "alignItems": "center",
-          "backgroundColor": "currentColor",
-          "border": "2px solid currentColor",
-          "borderRadius": 4,
-          "display": "flex",
+          "backgroundColor": "#A989C5",
+          "border": "2px solid #A989C5",
           "height": 16,
-          "justifyContent": "center",
           "width": 16,
         }
       }
@@ -320,40 +349,70 @@ exports[`StackedCheckBox should render "on inverted" correctly 1`] = `
       </svg>
     </div>
   </div>
-</span>
+</div>
 `;
 
-exports[`StackedCheckBox should render "on" correctly 1`] = `
-<span
-  className={undefined}
-  style={
-    Object {
-      "position": "relative",
-    }
-  }
+exports[`StackedCheckBox should render "Checked" correctly 1`] = `
+<div
+  className="relative"
 >
-  <div
-    className="cursor-pointer"
-    onClick={[Function]}
+  <span
+    className="absolute"
     style={
       Object {
-        "left": 3,
-        "position": "absolute",
-        "top": -3,
+        "left": 4,
+        "top": -4,
         "zIndex": -1,
       }
     }
   >
     <div
+      className="cursor-pointer"
+      onClick={[Function]}
+    >
+      <div
+        className="flex align-center justify-center rounded"
+        style={
+          Object {
+            "backgroundColor": "#509EE3",
+            "border": "2px solid #509EE3",
+            "height": 16,
+            "width": 16,
+          }
+        }
+      >
+        <svg
+          className="Icon Icon-check"
+          fill="currentcolor"
+          height={12}
+          name="check"
+          size={12}
+          style={
+            Object {
+              "color": "white",
+            }
+          }
+          viewBox="0 0 32 32"
+          width={12}
+        >
+          <path
+            d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
+          />
+        </svg>
+      </div>
+    </div>
+  </span>
+  <div
+    className="cursor-pointer"
+    onClick={[Function]}
+  >
+    <div
+      className="flex align-center justify-center rounded"
       style={
         Object {
-          "alignItems": "center",
-          "backgroundColor": "white",
-          "border": "2px solid #ddd",
-          "borderRadius": 4,
-          "display": "flex",
+          "backgroundColor": "#509EE3",
+          "border": "2px solid #509EE3",
           "height": 16,
-          "justifyContent": "center",
           "width": 16,
         }
       }
@@ -366,7 +425,7 @@ exports[`StackedCheckBox should render "on" correctly 1`] = `
         size={12}
         style={
           Object {
-            "color": "currentColor",
+            "color": "white",
           }
         }
         viewBox="0 0 32 32"
@@ -378,46 +437,57 @@ exports[`StackedCheckBox should render "on" correctly 1`] = `
       </svg>
     </div>
   </div>
+</div>
+`;
+
+exports[`StackedCheckBox should render "Off - Default" correctly 1`] = `
+<div
+  className="relative"
+>
+  <span
+    className="absolute"
+    style={
+      Object {
+        "left": 4,
+        "top": -4,
+        "zIndex": -1,
+      }
+    }
+  >
+    <div
+      className="cursor-pointer"
+      onClick={[Function]}
+    >
+      <div
+        className="flex align-center justify-center rounded"
+        style={
+          Object {
+            "backgroundColor": "white",
+            "border": "2px solid #ddd",
+            "height": 16,
+            "width": 16,
+          }
+        }
+      />
+    </div>
+  </span>
   <div
     className="cursor-pointer"
     onClick={[Function]}
-    style={Object {}}
   >
     <div
+      className="flex align-center justify-center rounded"
       style={
         Object {
-          "alignItems": "center",
           "backgroundColor": "white",
           "border": "2px solid #ddd",
-          "borderRadius": 4,
-          "display": "flex",
           "height": 16,
-          "justifyContent": "center",
           "width": 16,
         }
       }
-    >
-      <svg
-        className="Icon Icon-check"
-        fill="currentcolor"
-        height={12}
-        name="check"
-        size={12}
-        style={
-          Object {
-            "color": "currentColor",
-          }
-        }
-        viewBox="0 0 32 32"
-        width={12}
-      >
-        <path
-          d="M1 14 L5 10 L13 18 L27 4 L31 8 L13 26 z "
-        />
-      </svg>
-    </div>
+    />
   </div>
-</span>
+</div>
 `;
 
 exports[`Toggle should render "off" correctly 1`] = `
diff --git a/frontend/src/metabase/lib/api.js b/frontend/src/metabase/lib/api.js
index 75b9941e936a7e89611989d4c80fd92d3faf3317..ea661c351c4dc4053ee553b15525db6c6fcca066 100644
--- a/frontend/src/metabase/lib/api.js
+++ b/frontend/src/metabase/lib/api.js
@@ -82,8 +82,10 @@ class Api extends EventEmitter {
         }
     }
 
+    // TODO Atte Keinänen 6/26/17: Replacing this with isomorphic-fetch could simplify the implementation
     _makeRequest(method, url, headers, body, data, options) {
         return new Promise((resolve, reject) => {
+            let isCancelled = false;
             let xhr = new XMLHttpRequest();
             xhr.open(method, this.basename + url);
             for (let headerName in headers) {
@@ -102,7 +104,8 @@ class Api extends EventEmitter {
                     } else {
                         reject({
                             status: xhr.status,
-                            data: body
+                            data: body,
+                            isCancelled: isCancelled
                         });
                     }
                     if (!options.noEvent) {
@@ -113,10 +116,13 @@ class Api extends EventEmitter {
             xhr.send(body);
 
             if (options.cancelled) {
-                options.cancelled.then(() => xhr.abort());
+                options.cancelled.then(() => {
+                    isCancelled = true;
+                    xhr.abort()
+                });
             }
         });
     }
 }
 
-export default new Api();
+export default new Api();
\ No newline at end of file
diff --git a/frontend/src/metabase/lib/dom.js b/frontend/src/metabase/lib/dom.js
index af5a76cbd5017d7a543328d49755f8bbe1e490a4..8ff4c73b2607438aa58ce70654fc3e7a544c8bf2 100644
--- a/frontend/src/metabase/lib/dom.js
+++ b/frontend/src/metabase/lib/dom.js
@@ -231,3 +231,13 @@ export function moveToFront(element) {
         element.parentNode.appendChild(element);
     }
 }
+
+/**
+ * @returns the clip-path CSS property referencing the clip path in the current document, taking into account the <base> tag.
+ */
+export function clipPathReference(id: string): string {
+    // add the current page URL (with fragment removed) to support pages with <base> tag.
+    // https://stackoverflow.com/questions/18259032/using-base-tag-on-a-page-that-contains-svg-marker-elements-fails-to-render-marke
+    const url = window.location.href.replace(/#.*$/, "") + "#" + id;
+    return `url(${url})`;
+}
diff --git a/frontend/src/metabase/lib/query.js b/frontend/src/metabase/lib/query.js
index 6aafc089eab7a8c8f8a72f79ef0909dc38b39a0e..b0d790c8618d8e3057c8043eff95eadc4d47a832 100644
--- a/frontend/src/metabase/lib/query.js
+++ b/frontend/src/metabase/lib/query.js
@@ -334,17 +334,25 @@ var Query = {
         return Array.isArray(field) && mbqlEq(field[0], "aggregation");
     },
 
+    // field literal has the formal ["field-literal", <field-name>, <field-base-type>]
+    isFieldLiteral(field) {
+        return Array.isArray(field) && field.length === 3 && mbqlEq(field[0], "field-literal") && _.isString(field[1]) && _.isString(field[2]);
+    },
+
     isValidField(field) {
         return (
             (Query.isRegularField(field)) ||
             (Query.isLocalField(field)) ||
             (Query.isForeignKeyField(field) && Query.isRegularField(field[1]) && Query.isRegularField(field[2])) ||
+            // datetime field can  be either 4-item (deprecated): ["datetime-field", <field>, "as", <unit>]
+            // or 3 item (preferred style): ["datetime-field", <field>, <unit>]
             (Query.isDatetimeField(field)   && Query.isValidField(field[1]) &&
                 (field.length === 4 ?
                     (field[2] === "as" && typeof field[3] === "string") : // deprecated
                     typeof field[2] === "string")) ||
             (Query.isExpressionField(field) && _.isString(field[1])) ||
-            (Query.isAggregateField(field)  && typeof field[1] === "number")
+            (Query.isAggregateField(field)  && typeof field[1] === "number") ||
+            Query.isFieldLiteral(field)
         );
     },
 
@@ -368,6 +376,8 @@ var Query = {
             return Query.getFieldTargetId(field[1]);
         } else if (Query.isBinningStrategy(field)) {
             return Query.getFieldTargetId(field[1]);
+        } else if (Query.isFieldLiteral(field)) {
+            return field;
         }
         console.warn("Unknown field type: ", field);
     },
@@ -413,7 +423,9 @@ var Query = {
                 table: tableDef,
                 field: fieldDef,
                 path: path
-            }
+            };
+        } else if (Query.isFieldLiteral(field)) {
+            return { table: tableDef, field: Table.getField(tableDef, field), path }; // just pretend it's a normal field
         }
 
         console.warn("Unknown field type: ", field);
diff --git a/frontend/src/metabase/lib/query/field.js b/frontend/src/metabase/lib/query/field.js
index 9ddfd7e1390c5936cbfbc53a4c0805a2d6079288..ed5d95ad35f1c41f407ebbff8d2e439328816f26 100644
--- a/frontend/src/metabase/lib/query/field.js
+++ b/frontend/src/metabase/lib/query/field.js
@@ -21,6 +21,8 @@ export function getFieldTargetId(field: FieldReference): ?FieldId {
     } else if (isBinningStrategy(field)) {
         // $FlowFixMe
         return getFieldTargetId(field[1]);
+    } else if (isFieldLiteral(field)) {
+        return field;
     }
     console.warn("Unknown field type: ", field);
 }
@@ -45,6 +47,10 @@ export function isBinningStrategy(field: FieldReference): boolean {
     return Array.isArray(field) && mbqlEq(field[0], "binning-strategy");
 }
 
+export function isFieldLiteral(field: FieldReference): boolean {
+    return Array.isArray(field) && field.length === 3 && mbqlEq(field[0], "field-literal");
+}
+
 export function isExpressionField(field: FieldReference): boolean {
     return Array.isArray(field) && field.length === 2 && mbqlEq(field[0], "expression");
 }
diff --git a/frontend/src/metabase/lib/query/util.js b/frontend/src/metabase/lib/query/util.js
index 5c80c8130875e8d011f871ad34a773eac309f93c..806b6d3423fd27397de91a5925516b7e4522e524 100644
--- a/frontend/src/metabase/lib/query/util.js
+++ b/frontend/src/metabase/lib/query/util.js
@@ -8,6 +8,27 @@ export const mbql = (a: string):string =>
 export const mbqlEq = (a: string, b: string): boolean =>
     mbql(a) === mbql(b);
 
+// determines whether 2 field IDs are equal. This is needed rather than
+// doing a simple comparison because field IDs are not guaranteed to be numeric:
+// the might be FieldLiterals, e.g. [field-literal <name> <unit>], instead.
+export const fieldIdsEq = (a: any, b: any): boolean => {
+    if (typeof a !== typeof b) return false;
+
+    if (typeof a === "number") return a === b;
+
+    if (a == null && b == null) return true;
+
+    // field literals
+    if (Array.isArray(a) && Array.isArray(b) &&
+        a.length === 3 && b.length === 3 &&
+        a[0] === "field-literal" && b[0] === "field-literal") {
+        return a[1] === b[1];
+    }
+
+    console.warn("Don't know how to compare these IDs:", a, b);
+    return false;
+}
+
 export const noNullValues = (clause: any[]): boolean =>
     _.all(clause, c => c != null);
 
diff --git a/frontend/src/metabase/lib/query_time.js b/frontend/src/metabase/lib/query_time.js
index 9a56774d3ab4839d91c6f4ad20d288f67583cb15..335d9ab79fe9a2f96045fc18598de373e44fe9ba 100644
--- a/frontend/src/metabase/lib/query_time.js
+++ b/frontend/src/metabase/lib/query_time.js
@@ -183,14 +183,18 @@ export function parseFieldBucketing(field, defaultUnit = null) {
     if (Array.isArray(field)) {
         if (mbqlEq(field[0], "datetime-field")) {
             if (field.length === 4) {
-                // Deprecated legacy format, see DatetimeFieldDimension
+                // Deprecated legacy format [datetime-field field "as" unit], see DatetimeFieldDimension for more info
                 return field[3];
             } else {
+                // Current format [datetime-field field unit]
                 return field[2]
             }
         } if (mbqlEq(field[0], "fk->") || mbqlEq(field[0], "field-id")) {
             return defaultUnit;
-        } else {
+        } if (mbqlEq(field[0], "field-literal")) {
+            return defaultUnit;
+        }
+        else {
             console.warn("Unknown field format", field);
         }
     }
@@ -213,6 +217,7 @@ export function parseFieldTargetId(field) {
         if (mbqlEq(field[0], "field-id"))       return field[1];
         if (mbqlEq(field[0], "fk->"))           return field[1];
         if (mbqlEq(field[0], "datetime-field")) return parseFieldTargetId(field[1]);
+        if (mbqlEq(field[0], "field-literal"))  return field;
     }
 
     console.warn("Unknown field format", field);
diff --git a/frontend/src/metabase/lib/settings.js b/frontend/src/metabase/lib/settings.js
index 5460a7b95681cf17c26a8c03735ade94d01ef0f2..4c0fdfaefbcdeaa6d6d91ebc95a184cc27127efe 100644
--- a/frontend/src/metabase/lib/settings.js
+++ b/frontend/src/metabase/lib/settings.js
@@ -51,6 +51,10 @@ const MetabaseSettings = {
         return mb_settings.google_auth_client_id != null;
     },
 
+    ldapEnabled: function() {
+        return mb_settings.ldap_configured;
+    },
+
     newVersionAvailable: function(settings) {
         let versionInfo = _.findWhere(settings, {key: "version-info"}),
             currentVersion = MetabaseSettings.get("version").tag;
diff --git a/frontend/src/metabase/lib/urls.js b/frontend/src/metabase/lib/urls.js
index 3a83c230e064af2ca68e7209d85f102577e56c48..f60668536a4b224d0e8bb414c3845b2d0465e0fc 100644
--- a/frontend/src/metabase/lib/urls.js
+++ b/frontend/src/metabase/lib/urls.js
@@ -70,12 +70,12 @@ export function label(label) {
 }
 
 export function publicCard(uuid, type = null) {
-    const siteUrl = MetabaseSettings.get("site-url");
+    const siteUrl = MetabaseSettings.get("site_url");
     return `${siteUrl}/public/question/${uuid}` + (type ? `.${type}` : ``);
 }
 
 export function publicDashboard(uuid) {
-    const siteUrl = MetabaseSettings.get("site-url");
+    const siteUrl = MetabaseSettings.get("site_url");
     return `${siteUrl}/public/dashboard/${uuid}`;
 }
 
diff --git a/frontend/src/metabase/meta/Card.js b/frontend/src/metabase/meta/Card.js
index 54d3c6c7024daf6c59b08ea74f26e4e892dd85a8..3d73996f46bc47c99fce3d573156704a1e4d11d0 100644
--- a/frontend/src/metabase/meta/Card.js
+++ b/frontend/src/metabase/meta/Card.js
@@ -116,6 +116,8 @@ export function getParametersWithExtras(card: Card, parameterValues?: ParameterV
     })
 }
 
+// NOTE Atte Keinänen 7/5/17: Still used in dashboards and public questions.
+// Query builder uses `Question.getResults` which contains similar logic.
 export function applyParameters(
     card: Card,
     parameters: Parameter[],
diff --git a/frontend/src/metabase/meta/Dashboard.js b/frontend/src/metabase/meta/Dashboard.js
index eb4ebf035246e75a8fcde2b273559839b3af6a9e..6488d80f852e94c92770b5ca1fdcb529c255e15b 100644
--- a/frontend/src/metabase/meta/Dashboard.js
+++ b/frontend/src/metabase/meta/Dashboard.js
@@ -218,6 +218,7 @@ export function fieldFilterForParameterType(parameterType: ParameterType): Field
         case "id":          return (field: Field) => field.isID();
         case "category":    return (field: Field) => field.isCategory();
     }
+
     switch (parameterType) {
         case "location/city":     return (field: Field) => isa(field.special_type, TYPE.City);
         case "location/state":    return (field: Field) => isa(field.special_type, TYPE.State);
diff --git a/frontend/src/metabase/meta/types/Query.js b/frontend/src/metabase/meta/types/Query.js
index 831fceeba01a3955b281c1536efe850452c886a0..7be2639c27d0100bf081b69f12add8ce9ac05966 100644
--- a/frontend/src/metabase/meta/types/Query.js
+++ b/frontend/src/metabase/meta/types/Query.js
@@ -1,7 +1,7 @@
 /* @flow */
 
 import type { TableId } from "./Table";
-import type { FieldId } from "./Field";
+import type { FieldId, BaseType } from "./Field";
 import type { SegmentId } from "./Segment";
 import type { MetricId } from "./Metric";
 import type { ParameterType } from "./Parameter";
@@ -156,6 +156,9 @@ export type ForeignFieldReference =
 export type ExpressionReference =
     ["expression", ExpressionName];
 
+export type FieldLiteral =
+    ["field-literal", string, BaseType]; // ["field-literal", name, base-type]
+
 export type DatetimeField =
     ["datetime-field", LocalFieldReference | ForeignFieldReference, DatetimeUnit] |
     ["datetime-field", LocalFieldReference | ForeignFieldReference, "as", DatetimeUnit]; // @deprecated: don't include the "as" element
diff --git a/frontend/src/metabase/meta/types/Visualization.js b/frontend/src/metabase/meta/types/Visualization.js
index 3f302805054f249795d92f085434957056acf410..3e939d2de50bad825997db8f681bfb3d8e166647 100644
--- a/frontend/src/metabase/meta/types/Visualization.js
+++ b/frontend/src/metabase/meta/types/Visualization.js
@@ -51,8 +51,10 @@ export type ClickActionProps = {
     clicked?: ClickObject
 }
 
+export type OnChangeCardAndRun = ({ nextCard: Card, previousCard?: ?Card }) => void
+
 export type ClickActionPopoverProps = {
-    onChangeCardAndRun: (Object) => void,
+    onChangeCardAndRun: OnChangeCardAndRun,
     onClose: () => void,
 }
 
@@ -76,11 +78,16 @@ export type VisualizationProps = {
     isEditing: boolean,
     actionButtons: Node,
 
+    onRender: ({
+        yAxisSplit?: number[][],
+        warnings?: string[]
+    }) => void,
+
     hovered: ?HoverObject,
     onHoverChange: (?HoverObject) => void,
     onVisualizationClick: (?ClickObject) => void,
     visualizationIsClickable: (?ClickObject) => boolean,
-    onChangeCardAndRun: (Object) => void,
+    onChangeCardAndRun: OnChangeCardAndRun,
 
     onUpdateVisualizationSettings: ({ [key: string]: any }) => void
 }
diff --git a/frontend/src/metabase/nav/components/ProfileLink.jsx b/frontend/src/metabase/nav/components/ProfileLink.jsx
index d9b3ad48a7087e8c4656980fdad314394150a644..6c8c4694e7259871be97917a661e8a2bd25084eb 100644
--- a/frontend/src/metabase/nav/components/ProfileLink.jsx
+++ b/frontend/src/metabase/nav/components/ProfileLink.jsx
@@ -77,11 +77,13 @@ export default class ProfileLink extends Component {
                     <OnClickOutsideWrapper handleDismissal={this.closeDropdown}>
                         <div className="NavDropdown-content right">
                             <ul className="NavDropdown-content-layer">
-                                <li>
-                                    <Link to="/user/edit_current" data-metabase-event={"Navbar;Profile Dropdown;Edit Profile"} onClick={this.closeDropdown} className="Dropdown-item block text-white no-decoration">
-                                        Account Settings
-                                    </Link>
-                                </li>
+                                { !user.google_auth && !user.ldap_auth ?
+                                    <li>
+                                        <Link to="/user/edit_current" data-metabase-event={"Navbar;Profile Dropdown;Edit Profile"} onClick={this.closeDropdown} className="Dropdown-item block text-white no-decoration">
+                                            Account Settings
+                                        </Link>
+                                    </li>
+                                : null }
 
                                 { user.is_superuser && context !== 'admin' ?
                                     <li>
diff --git a/frontend/src/metabase/qb/components/TimeseriesFilterWidget.jsx b/frontend/src/metabase/qb/components/TimeseriesFilterWidget.jsx
index ed6e0296717c99c73e7c8a1404778a1522441df9..90b248396a23e3934b75a74567585b336d947c41 100644
--- a/frontend/src/metabase/qb/components/TimeseriesFilterWidget.jsx
+++ b/frontend/src/metabase/qb/components/TimeseriesFilterWidget.jsx
@@ -106,6 +106,10 @@ export default class TimeseriesFilterWidget extends Component {
                 currentDescription = "After " + currentDescription;
             } else if (currentFilter[0] === "<") {
                 currentDescription = "Before " + currentDescription;
+            } else if (currentFilter[0] === "IS_NULL") {
+                currentDescription = "Is Empty";
+            } else if (currentFilter[0] === "NOT_NULL") {
+                currentDescription = "Not Empty";
             }
         } else {
             currentDescription = "All Time";
diff --git a/frontend/src/metabase/qb/components/TimeseriesFilterWidget.spec.jsx b/frontend/src/metabase/qb/components/TimeseriesFilterWidget.spec.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..f5856207278b16434a743a0de6f42a2131bbeafa
--- /dev/null
+++ b/frontend/src/metabase/qb/components/TimeseriesFilterWidget.spec.jsx
@@ -0,0 +1,64 @@
+/* eslint-disable flowtype/require-valid-file-annotation */
+import React from "react";
+import TimeseriesFilterWidget from "./TimeseriesFilterWidget";
+import { mount } from "enzyme";
+
+import Question from "metabase-lib/lib/Question";
+import {
+    DATABASE_ID,
+    ORDERS_TABLE_ID,
+    metadata
+} from "metabase/__support__/sample_dataset_fixture";
+
+const getTimeseriesFilterWidget = question => (
+    <TimeseriesFilterWidget
+        card={question.card()}
+        tableMetadata={question.tableMetadata()}
+        datasetQuery={question.query().datasetQuery()}
+        setDatasetQuery={() => {}}
+    />
+);
+
+describe("TimeseriesFilterWidget", () => {
+    const questionWithoutFilter = Question.create({
+        databaseId: DATABASE_ID,
+        tableId: ORDERS_TABLE_ID,
+        metadata
+    })
+        .query()
+        .addAggregation(["count"])
+        .addBreakout(["datetime-field", ["field-id", 1], "day"])
+        .question();
+
+    it("should display 'All Time' text if no filter is selected", () => {
+        const widget = mount(getTimeseriesFilterWidget(questionWithoutFilter));
+        expect(widget.find(".AdminSelect-content").text()).toBe("All Time");
+    });
+    it("should display 'Past 30 Days' text if that filter is selected", () => {
+        const questionWithFilter = questionWithoutFilter
+            .query()
+            .addFilter(["time-interval", ["field-id", 1], -30, "day"])
+            .question();
+
+        const widget = mount(getTimeseriesFilterWidget(questionWithFilter));
+        expect(widget.find(".AdminSelect-content").text()).toBe("Past 30 Days");
+    });
+    it("should display 'Is Empty' text if that filter is selected", () => {
+        const questionWithFilter = questionWithoutFilter
+            .query()
+            .addFilter(["IS_NULL", ["field-id", 1]])
+            .question();
+
+        const widget = mount(getTimeseriesFilterWidget(questionWithFilter));
+        expect(widget.find(".AdminSelect-content").text()).toBe("Is Empty");
+    });
+    it("should display 'Not Empty' text if that filter is selected", () => {
+        const questionWithFilter = questionWithoutFilter
+            .query()
+            .addFilter(["NOT_NULL", ["field-id", 1]])
+            .question();
+
+        const widget = mount(getTimeseriesFilterWidget(questionWithFilter));
+        expect(widget.find(".AdminSelect-content").text()).toBe("Not Empty");
+    });
+});
diff --git a/frontend/src/metabase/qb/components/actions/PivotByAction.jsx b/frontend/src/metabase/qb/components/actions/PivotByAction.jsx
index 5a1e852671e375ae99a58aef0e921325314fed35..20f3d8e7b2c14efba23b59a5a6a2a6861c4b0d49 100644
--- a/frontend/src/metabase/qb/components/actions/PivotByAction.jsx
+++ b/frontend/src/metabase/qb/components/actions/PivotByAction.jsx
@@ -65,11 +65,13 @@ export default (name: string, icon: string, fieldFilter: FieldFilter) =>
                         tableMetadata={tableMetadata}
                         fieldOptions={breakoutOptions}
                         onCommitBreakout={breakout => {
-                            onChangeCardAndRun({
-                                nextCard: question
-                                    .pivot([breakout], dimensions)
-                                    .card()
-                            });
+                            const nextCard = question
+                                .pivot([breakout], dimensions)
+                                .card();
+
+                            if (nextCard) {
+                                onChangeCardAndRun({ nextCard });
+                            }
                         }}
                         onClose={onClose}
                     />
diff --git a/frontend/src/metabase/qb/components/actions/SummarizeBySegmentMetricAction.unit.spec.js b/frontend/src/metabase/qb/components/actions/SummarizeBySegmentMetricAction.unit.spec.js
index 7521ce53cb482d69cbc09afe5f07471165dd157c..d8489c8b28e882a873b926f3457824e8fcf3b90f 100644
--- a/frontend/src/metabase/qb/components/actions/SummarizeBySegmentMetricAction.unit.spec.js
+++ b/frontend/src/metabase/qb/components/actions/SummarizeBySegmentMetricAction.unit.spec.js
@@ -52,7 +52,6 @@ describe("SummarizeBySegmentMetricAction", () => {
                 component
                     .find('.List-item-title[children="Sum of ..."]')
                     .simulate("click");
-                console.log(component.debug());
 
                 component
                     .find('.List-item-title[children="Subtotal"]')
diff --git a/frontend/src/metabase/qb/components/drill/ObjectDetailDrill.unit.spec.js b/frontend/src/metabase/qb/components/drill/ObjectDetailDrill.unit.spec.js
index 784d0d321069ced14f16bf3552df15936830330b..d7daec03df3e855c84a36efd170f65cec555248d 100644
--- a/frontend/src/metabase/qb/components/drill/ObjectDetailDrill.unit.spec.js
+++ b/frontend/src/metabase/qb/components/drill/ObjectDetailDrill.unit.spec.js
@@ -33,7 +33,6 @@ describe("ObjectDetailDrill", () => {
             source_table: ORDERS_TABLE_ID,
             filter: ["=", ["field-id", ORDERS_PK_FIELD_ID], 42]
         });
-        expect(newCard.display).toEqual(undefined);
     });
     it("should be return correct new card for FKs", () => {
         const actions = ObjectDetailDrill({
@@ -46,6 +45,5 @@ describe("ObjectDetailDrill", () => {
             source_table: PRODUCT_TABLE_ID,
             filter: ["=", ["field-id", PRODUCT_PK_FIELD_ID], 43]
         });
-        expect(newCard.display).toEqual(undefined);
     });
 });
diff --git a/frontend/src/metabase/qb/components/drill/TimeseriesPivotDrill.integ.spec.js b/frontend/src/metabase/qb/components/drill/TimeseriesPivotDrill.integ.spec.js
index d1fa3ccd7a8f00ba557bc105b16715781ede2019..374ff7274f87011938759e3a9f9ed3e37c1e7b10 100644
--- a/frontend/src/metabase/qb/components/drill/TimeseriesPivotDrill.integ.spec.js
+++ b/frontend/src/metabase/qb/components/drill/TimeseriesPivotDrill.integ.spec.js
@@ -4,4 +4,9 @@ describe("TimeseriesPivotDrill", () => {
         // Would be nice to run an integration test here to make sure that the resulting MBQL is valid and runnable
         pending();
     });
+    it("should accept the dimension value as a year string as in table visualization", () => {
+        // Intented to test that "Zoom In" for a table cell when you have broken out by year works correctly
+        // Could also be part of more comprehensive QB integrated test where the table cell is actually clicked
+        pending();
+    });
 });
diff --git a/frontend/src/metabase/qb/lib/actions.js b/frontend/src/metabase/qb/lib/actions.js
index 091026fc75df155b426d8440ba524e434f95f7ca..eef6b34227a46fc81df42748dcc888f5a87f3e81 100644
--- a/frontend/src/metabase/qb/lib/actions.js
+++ b/frontend/src/metabase/qb/lib/actions.js
@@ -4,6 +4,7 @@ import moment from "moment";
 import _ from "underscore";
 
 import Q from "metabase/lib/query"; // legacy query lib
+import { fieldIdsEq } from "metabase/lib/query/util";
 import * as Card from "metabase/meta/Card";
 import * as Query from "metabase/lib/query/query";
 import * as Field from "metabase/lib/query/field";
@@ -17,6 +18,7 @@ import type Table from "metabase-lib/lib/metadata/Table";
 import type { Card as CardObject } from "metabase/meta/types/Card";
 import type { StructuredQuery, FieldFilter } from "metabase/meta/types/Query";
 import type { DimensionValue } from "metabase/meta/types/Visualization";
+import { parseTimestamp } from "metabase/lib/time";
 
 // TODO: use icepick instead of mutation, make they handle frozen cards
 
@@ -94,7 +96,7 @@ const drillFilter = (card, value, column) => {
                 "as",
                 column.unit
             ],
-            moment(value).toISOString()
+            parseTimestamp(value, column.unit).toISOString()
         ];
     } else {
         const range = rangeForValue(value, column);
@@ -141,8 +143,10 @@ export const addOrUpdateBreakout = (card, breakout) => {
     let breakouts = Query.getBreakouts(newCard.dataset_query.query);
     for (let index = 0; index < breakouts.length; index++) {
         if (
-            Field.getFieldTargetId(breakouts[index]) ===
-            Field.getFieldTargetId(breakout)
+            fieldIdsEq(
+                Field.getFieldTargetId(breakouts[index]),
+                Field.getFieldTargetId(breakout)
+            )
         ) {
             newCard.dataset_query.query = Query.updateBreakout(
                 newCard.dataset_query.query,
@@ -217,7 +221,7 @@ export const breakout = (card, breakout, tableMetadata) => {
 // min number of points when switching units
 const MIN_INTERVALS = 4;
 
-export const updateDateTimeFilter = (card, column, start, end) => {
+export const updateDateTimeFilter = (card, column, start, end): CardObject => {
     let newCard = clone(card);
 
     let fieldRef = getFieldRefFromColumn(column);
@@ -312,7 +316,7 @@ export const pivot = (
             tableMetadata
         );
         for (const [index, field] of breakoutFields.entries()) {
-            if (field && field.id === dimension.column.id) {
+            if (field && fieldIdsEq(field.id, dimension.column.id)) {
                 newCard.dataset_query.query = Query.removeBreakout(
                     newCard.dataset_query.query,
                     index
diff --git a/frontend/src/metabase/query_builder/actions.integ.spec.js b/frontend/src/metabase/query_builder/actions.integ.spec.js
index 1eca38457c505bf08b5f8d8e0abe29d81585846d..81e9592d563e8e4b8a453eebbf429065665d28da 100644
--- a/frontend/src/metabase/query_builder/actions.integ.spec.js
+++ b/frontend/src/metabase/query_builder/actions.integ.spec.js
@@ -1,64 +1,50 @@
 import {
-    DATABASE_ID, ORDERS_TABLE_ID, metadata,
-    ORDERS_TOTAL_FIELD_ID
+    ORDERS_TOTAL_FIELD_ID,
+    unsavedOrderCountQuestion
 } from "metabase/__support__/sample_dataset_fixture";
 import Question from "metabase-lib/lib/Question";
 import { parse as urlParse } from "url";
 import {
-    login,
-    globalReduxStore as store,
-    globalBrowserHistory as history
+    createSavedQuestion,
+    createTestStore,
+    login
 } from "metabase/__support__/integrated_tests";
 import { initializeQB } from "./actions";
 import { getCard, getOriginalCard, getQueryResults } from "./selectors";
-import { CardApi } from "metabase/services";
-import { refreshSiteSettings } from "metabase/redux/settings";
+import _ from "underscore";
 
 jest.mock('metabase/lib/analytics');
 
+// TODO: Convert completely to modern style
 
 describe("QueryBuilder", () => {
-    let unsavedQuestion: Question = null;
     let savedCleanQuestion: Question = null;
     let dirtyQuestion: Question = null;
+    let store = null;
 
     beforeAll(async () => {
         await login();
+        store = await createTestStore()
     })
 
     describe("initializeQb", () => {
         beforeAll(async () => {
-            // Question initialization has to be in beforeAll block due to the CardApi.create api call
-            await store.dispatch(refreshSiteSettings());
-
-            unsavedQuestion = Question.create({databaseId: DATABASE_ID, tableId: ORDERS_TABLE_ID, metadata})
-                .query()
-                .addAggregation(["count"])
-                .question()
-
-            unsavedQuestion._card = { ...unsavedQuestion._card, name: "Order count" }
-
-            const savedCleanQuestionCard = await CardApi.create(unsavedQuestion.card())
-            savedCleanQuestion = unsavedQuestion.setCard({
-                ...savedCleanQuestionCard
-            });
+            savedCleanQuestion = await createSavedQuestion(unsavedOrderCountQuestion)
 
             dirtyQuestion = savedCleanQuestion
                 .query()
                 .addBreakout(["field-id", ORDERS_TOTAL_FIELD_ID])
                 .question()
-
-            dirtyQuestion._card = { ...dirtyQuestion._card, original_card_id: dirtyQuestion.id() }
         })
 
         describe("for unsaved questions", () => {
             it("completes successfully", async () => {
-                const location = urlParse(unsavedQuestion.getUrl())
+                const location = urlParse(unsavedOrderCountQuestion.getUrl())
                 await store.dispatch(initializeQB(location, {}))
             });
 
             it("results in the correct card object in redux state", async () => {
-                expect(getCard(store.getState())).toMatchObject(unsavedQuestion.card())
+                expect(getCard(store.getState())).toMatchObject(unsavedOrderCountQuestion.card())
             })
 
             it("results in an empty original_card object in redux state", async () => {
@@ -66,11 +52,7 @@ describe("QueryBuilder", () => {
             })
 
             it("keeps the url same after initialization is finished", async () => {
-                const location = urlParse(unsavedQuestion.getUrl())
-
-                // can't get the location from Redux state for some reason so just query the emulated history object directly
-                expect(history.getCurrentLocation().pathname).toBe(location.pathname)
-                expect(history.getCurrentLocation().hash).toBe(location.hash)
+                expect(store.getPath()).toBe(unsavedOrderCountQuestion.getUrl())
             })
 
             // TODO: setTimeout for
@@ -87,18 +69,14 @@ describe("QueryBuilder", () => {
                 });
 
                 it("results in the correct card object in redux state", async () => {
-                    expect(getCard(store.getState())).toMatchObject(savedCleanQuestion.card())
+                    expect(getCard(store.getState())).toMatchObject(_.omit(savedCleanQuestion.card(), "original_card_id"))
                 })
 
                 it("results in the correct original_card object in redux state", async () => {
-                    expect(getOriginalCard(store.getState())).toMatchObject(savedCleanQuestion.card())
+                    expect(getOriginalCard(store.getState())).toMatchObject(_.omit(savedCleanQuestion.card(), "original_card_id"))
                 })
                 it("keeps the url same after initialization is finished", async () => {
-                    const location = urlParse(savedCleanQuestion.getUrl(savedCleanQuestion))
-
-                    // can't get the location from Redux state for some reason so just query the emulated history object directly
-                    expect(history.getCurrentLocation().pathname).toBe(location.pathname)
-                    expect(history.getCurrentLocation().hash).toBe(location.hash || "")
+                    expect(store.getPath()).toBe(savedCleanQuestion.getUrl(savedCleanQuestion))
                 })
             })
             describe("with dirty state", () => {
@@ -112,14 +90,10 @@ describe("QueryBuilder", () => {
                 })
 
                 it("results in the correct original_card object in redux state", async () => {
-                    expect(getOriginalCard(store.getState())).toMatchObject(savedCleanQuestion.card())
+                    expect(getOriginalCard(store.getState())).toMatchObject(_.omit(savedCleanQuestion.card(), "original_card_id"))
                 })
                 it("keeps the url same after initialization is finished", async () => {
-                    const location = urlParse(dirtyQuestion.getUrl(savedCleanQuestion))
-
-                    // can't get the location from Redux state for some reason so just query the emulated history object directly
-                    expect(history.getCurrentLocation().pathname).toBe(location.pathname)
-                    expect(history.getCurrentLocation().hash).toBe(location.hash || "")
+                    expect(store.getPath()).toBe(dirtyQuestion.getUrl())
                 })
             })
         })
diff --git a/frontend/src/metabase/query_builder/actions.js b/frontend/src/metabase/query_builder/actions.js
index d90341fd78da9f59dffc38960b3be58fdba9e2b9..a8867412db9b7e3b42dfafcff32bcd7990348cb3 100644
--- a/frontend/src/metabase/query_builder/actions.js
+++ b/frontend/src/metabase/query_builder/actions.js
@@ -29,7 +29,8 @@ import {
     getQuestion,
     getOriginalQuestion,
     getOriginalCard,
-    getIsEditing
+    getIsEditing,
+    getIsShowingDataReference
 } from "./selectors";
 
 import { getDatabases, getTables, getDatabasesList, getMetadata } from "metabase/selectors/metadata";
@@ -236,7 +237,7 @@ export const initializeQB = (location, params) => {
 
                 preserveParameters = true;
             } catch(error) {
-                console.warn(error);
+                console.warn('initializeQb failed because of an error:', error);
                 card = null;
                 dispatch(setErrorPage(error));
             }
@@ -250,8 +251,8 @@ export const initializeQB = (location, params) => {
 
         } else {
             // we are starting a new/empty card
-
-            card = startNewCard("query");
+            const databaseId = (options.db) ? parseInt(options.db) : undefined;
+            card = startNewCard("query", databaseId);
 
             // initialize parts of the query based on optional parameters supplied
             if (options.table != undefined && card.dataset_query.query) {
@@ -279,7 +280,7 @@ export const initializeQB = (location, params) => {
         });
 
         // Fetch the question metadata
-        dispatch(loadMetadataForCard(card));
+        card && dispatch(loadMetadataForCard(card));
 
         const question = card && new Question(getMetadata(getState()), card);
 
@@ -368,18 +369,18 @@ export const loadMetadataForCard = createThunkAction(LOAD_METADATA_FOR_CARD, (ca
 
         const query = card && new Question(getMetadata(getState()), card).query();
 
-        function loadMetadataForAtomicQuery(singleQuery) {
+        async function loadMetadataForAtomicQuery(singleQuery) {
             if (singleQuery instanceof StructuredQuery && singleQuery.tableId() != null) {
-                dispatch(loadTableMetadata(singleQuery.tableId()));
+                await dispatch(loadTableMetadata(singleQuery.tableId()));
             }
 
             if (singleQuery instanceof NativeQuery && singleQuery.databaseId() != null) {
-                dispatch(loadDatabaseFields(singleQuery.databaseId()));
+                await dispatch(loadDatabaseFields(singleQuery.databaseId()));
             }
         }
 
         if (query) {
-            loadMetadataForAtomicQuery(query);
+            await loadMetadataForAtomicQuery(query);
         }
     }
 });
@@ -399,6 +400,7 @@ export const loadTableMetadata = createThunkAction(LOAD_TABLE_METADATA, (tableId
     };
 });
 
+// TODO Atte Keinänen 7/5/17: Move the API call to redux/metadata for being able to see the db fields in the new metadata object
 export const LOAD_DATABASE_FIELDS = "metabase/qb/LOAD_DATABASE_FIELDS";
 export const loadDatabaseFields = createThunkAction(LOAD_DATABASE_FIELDS, (dbId) => {
     return async (dispatch, getState) => {
@@ -618,14 +620,13 @@ export const updateQuestion = (newQuestion) => {
         dispatch.action(UPDATE_QUESTION, { card: newQuestion.card() });
 
         // See if the template tags editor should be shown/hidden
-
         const oldQuestion = getQuestion(getState());
         const oldTagCount = getTemplateTagCount(oldQuestion);
         const newTagCount = getTemplateTagCount(newQuestion);
 
         if (newTagCount > oldTagCount) {
             dispatch(setIsShowingTemplateTagsEditor(true));
-        } else if (newTagCount === 0) {
+        } else if (newTagCount === 0 && !getIsShowingDataReference(getState())) {
             dispatch(setIsShowingTemplateTagsEditor(false));
         }
     };
@@ -1018,7 +1019,7 @@ export const queryCompleted = createThunkAction(QUERY_COMPLETED, (card, queryRes
 export const QUERY_ERRORED = "metabase/qb/QUERY_ERRORED";
 export const queryErrored = createThunkAction(QUERY_ERRORED, (startTime, error) => {
     return async (dispatch, getState) => {
-        if (error && error.status === 0) {
+        if (error && error.isCancelled) {
             // cancelled, do nothing
             return null;
         } else {
@@ -1031,10 +1032,10 @@ export const queryErrored = createThunkAction(QUERY_ERRORED, (startTime, error)
 export const CANCEL_QUERY = "metabase/qb/CANCEL_QUERY";
 export const cancelQuery = createThunkAction(CANCEL_QUERY, () => {
     return async (dispatch, getState) => {
-        const { qb: { uiControls, queryExecutionPromise } } = getState();
+        const { qb: { uiControls, cancelQueryDeferred } } = getState();
 
-        if (uiControls.isRunning && queryExecutionPromise) {
-            queryExecutionPromise.resolve();
+        if (uiControls.isRunning && cancelQueryDeferred) {
+            cancelQueryDeferred.resolve();
         }
     };
 });
diff --git a/frontend/src/metabase/query_builder/components/DataSelector.jsx b/frontend/src/metabase/query_builder/components/DataSelector.jsx
index 1b8a48983a2c2636121fdced07ad8a41abc90b8d..2e30b09fe27874bbc1f821b301d8122ced7a111a 100644
--- a/frontend/src/metabase/query_builder/components/DataSelector.jsx
+++ b/frontend/src/metabase/query_builder/components/DataSelector.jsx
@@ -12,17 +12,15 @@ import _ from "underscore";
 import cx from "classnames";
 
 export default class DataSelector extends Component {
-    constructor(props, context) {
-        super(props, context);
 
+    constructor(props) {
+        super()
         this.state = {
             databases: null,
             selectedSchema: null,
             showTablePicker: true,
             showSegmentPicker: props.segments && props.segments.length > 0
         }
-
-        _.bindAll(this, "onChangeDatabase", "onChangeSchema", "onChangeTable", "onChangeSegment", "onBack");
     }
 
     static propTypes = {
@@ -84,7 +82,7 @@ export default class DataSelector extends Component {
         }
     }
 
-    onChangeTable(item) {
+    onChangeTable = (item) => {
         if (item.table != null) {
             this.props.setSourceTableFn(item.table.id);
         } else if (item.database != null) {
@@ -93,7 +91,7 @@ export default class DataSelector extends Component {
         this.refs.popover.toggle();
     }
 
-    onChangeSegment(item) {
+    onChangeSegment = (item) => {
         if (item.segment != null) {
             this.props.setSourceSegmentFn(item.segment.id);
         }
@@ -101,27 +99,27 @@ export default class DataSelector extends Component {
         this.refs.popover.toggle();
     }
 
-    onChangeSchema(schema) {
+    onChangeSchema = (schema) => {
         this.setState({
             selectedSchema: schema,
             showTablePicker: true
         });
     }
 
-    onChangeSegmentSection() {
+    onChangeSegmentSection = () => {
         this.setState({
             showSegmentPicker: true
         });
     }
 
-    onBack() {
+    onBack = () => {
         this.setState({
             showTablePicker: false,
             showSegmentPicker: false
         });
     }
 
-    onChangeDatabase(index) {
+    onChangeDatabase = (index) => {
         let database = this.state.databases[index];
         let schema = database && (database.schemas.length > 1 ? null : database.schemas[0]);
         if (database && database.tables.length === 0) {
@@ -172,35 +170,59 @@ export default class DataSelector extends Component {
     }
 
     renderDatabaseSchemaPicker() {
-        const { selectedSchema } = this.state;
-
-        let sections = this.state.databases.map(database => {
-            return {
+        const { databases, selectedSchema } = this.state;
+
+        let sections = databases
+            .filter(database =>
+                // filter out the saved questions "db" so we can present it
+                // differently
+                !database.is_saved_questions
+            )
+            .map(database => ({
                 name: database.name,
                 items: database.schemas.length > 1 ? database.schemas : []
-            };
-        });
+            }));
+
+        // do the opposite of what we just did and get a reference to the saved question "db"
+        // there will only ever be one of these hence [0]
+        const savedQuestionSection = databases.filter(db => db.is_saved_questions)[0]
+
+        // some of the change functions need the index in the databases array
+        const savedQuestionSectionIndex = databases.indexOf(savedQuestionSection)
 
-        let openSection = selectedSchema && _.findIndex(this.state.databases, (db) => _.find(db.schemas, selectedSchema));
-        if (openSection >= 0 && this.state.databases[openSection] && this.state.databases[openSection].schemas.length === 1) {
+        let openSection = selectedSchema && _.findIndex(databases, (db) => _.find(db.schemas, selectedSchema));
+        if (openSection >= 0 && databases[openSection] && databases[openSection].schemas.length === 1) {
             openSection = -1;
         }
 
         return (
-            <AccordianList
-                id="DatabaseSchemaPicker"
-                key="schemaPicker"
-                className="text-brand"
-                sections={sections}
-                onChange={this.onChangeSchema}
-                onChangeSection={this.onChangeDatabase}
-                itemIsSelected={(schema) => this.state.selectedSchema === schema}
-                renderSectionIcon={() => <Icon className="Icon text-default" name="database" size={18} />}
-                renderItemIcon={() => <Icon name="folder" size={16} />}
-                initiallyOpenSection={openSection}
-                showItemArrows={true}
-                alwaysTogglable={true}
-            />
+            <div>
+                <AccordianList
+                    id="DatabaseSchemaPicker"
+                    key="schemaPicker"
+                    className="text-brand"
+                    sections={sections}
+                    onChange={this.onChangeSchema}
+                    onChangeSection={this.onChangeDatabase}
+                    itemIsSelected={(schema) => this.state.selectedSchema === schema}
+                    renderSectionIcon={() => <Icon className="Icon text-default" name="database" size={18} />}
+                    renderItemIcon={() => <Icon name="folder" size={16} />}
+                    initiallyOpenSection={openSection}
+                    showItemArrows={true}
+                    alwaysTogglable={true}
+                />
+                { savedQuestionSection && (
+                    <div
+                        className="List-section p2 cursor-pointer text-brand-hover bg-slate-extra-light"
+                        onClick={() => this.onChangeDatabase(savedQuestionSectionIndex)}
+                    >
+                        <div className="List-section-header flex align-center">
+                            <Icon className="Icon text-default mr2" size={18} name="all" />
+                            <h3 className="List-section-title">Saved questions</h3>
+                        </div>
+                    </div>
+                )}
+            </div>
         );
     }
 
@@ -245,6 +267,8 @@ export default class DataSelector extends Component {
     renderTablePicker() {
         const schema = this.state.selectedSchema;
 
+        const isSavedQuestionList = schema.database.is_saved_questions;
+
         const hasMultipleDatabases = this.props.databases.length > 1;
         const hasMultipleSchemas = schema && schema.database && _.uniq(schema.database.tables, (t) => t.schema).length > 1;
         const hasSegments = !!this.props.segments;
@@ -252,7 +276,7 @@ export default class DataSelector extends Component {
 
         let header = (
             <span className="flex align-center">
-                <span className={cx("flex align-center text-slate", { "cursor-pointer": hasMultipleSources })} onClick={hasMultipleSources && this.onBack}>
+                <span className={cx("flex align-center text-brand-hover text-slate", { "cursor-pointer": hasMultipleSources })} onClick={hasMultipleSources && this.onBack}>
                     { hasMultipleSources && <Icon name="chevronleft" size={18} /> }
                     <span className="ml1">{schema.database.name}</span>
                 </span>
@@ -265,7 +289,7 @@ export default class DataSelector extends Component {
         if (schema.tables.length === 0) {
             // this is a database with no tables!
             return (
-                <section className="List-section List-section--open" style={{width: '300px'}}>
+                <section className="List-section List-section--open" style={{width: 300}}>
                     <div className="p1 border-bottom">
                         <div className="px1 py1 flex align-center">
                             <h3 className="text-default">{header}</h3>
@@ -286,18 +310,26 @@ export default class DataSelector extends Component {
                     }))
             }];
             return (
-                <AccordianList
-                    id="TablePicker"
-                    key="tablePicker"
-                    className="text-brand"
-                    sections={sections}
-                    searchable={true}
-                    onChange={this.onChangeTable}
-                    itemIsSelected={(item) => item.table ? item.table.id === this.getTableId() : false}
-                    itemIsClickable={(item) => item.table && !item.disabled}
-                    renderItemIcon={(item) => item.table ? <Icon name="table2" size={18} /> : null}
-                    hideSingleSectionTitle={true}
-                />
+                <div style={{ width: 300 }}>
+                    <AccordianList
+                        id="TablePicker"
+                        key="tablePicker"
+                        className="text-brand"
+                        sections={sections}
+                        searchable={true}
+                        onChange={this.onChangeTable}
+                        itemIsSelected={(item) => item.table ? item.table.id === this.getTableId() : false}
+                        itemIsClickable={(item) => item.table && !item.disabled}
+                        renderItemIcon={(item) => item.table ? <Icon name="table2" size={18} /> : null}
+                        hideSingleSectionTitle={true}
+                    />
+                    { isSavedQuestionList && (
+                        <div className="bg-slate-extra-light p2 text-centered">
+                            Is a question missing?
+                            <a href="http://metabase.com/docs/latest/users-guide/04-asking-questions.html#source-data" className="block link">Learn more about nested queries</a>
+                        </div>
+                    )}
+                </div>
             );
         }
     }
diff --git a/frontend/src/metabase/query_builder/components/FieldList.integ.spec.js b/frontend/src/metabase/query_builder/components/FieldList.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..c82ffaee9251f7103f6c6ed3abd851a902d4c87a
--- /dev/null
+++ b/frontend/src/metabase/query_builder/components/FieldList.integ.spec.js
@@ -0,0 +1,85 @@
+// Important: import of integrated_tests always comes first in tests because of mocked modules
+import { createTestStore, login } from "metabase/__support__/integrated_tests";
+
+import React from 'react'
+import { mount } from 'enzyme'
+
+import FieldList from './FieldList';
+import Question from "metabase-lib/lib/Question";
+import {
+    DATABASE_ID,
+    ORDERS_TABLE_ID,
+    orders_past_30_days_segment
+} from "metabase/__support__/sample_dataset_fixture";
+
+import StructuredQuery from "metabase-lib/lib/queries/StructuredQuery";
+import { createSegment } from "metabase/admin/datamodel/datamodel";
+import { getMetadata } from "metabase/selectors/metadata";
+import { fetchDatabases, fetchSegments, fetchTableMetadata } from "metabase/redux/metadata";
+import { TestTooltip, TestTooltipContent } from "metabase/components/Tooltip";
+import FilterWidget from "metabase/query_builder/components/filters/FilterWidget";
+
+const getFieldList = (query, fieldOptions, segmentOptions) =>
+    <FieldList
+        tableMetadata={query.tableMetadata()}
+        fieldOptions={fieldOptions}
+        segmentOptions={segmentOptions}
+        customFieldOptions={query.expressions()}
+        onFieldChange={() => {}}
+        enableSubDimensions={false}
+    />;
+
+describe('FieldList', () => {
+    beforeAll(async () => {
+        await login();
+    })
+
+    it("should allow using expression as aggregation dimension", async () => {
+        const store = await createTestStore()
+        await store.dispatch(fetchDatabases());
+        await store.dispatch(fetchTableMetadata(ORDERS_TABLE_ID));
+
+        const expressionName = "70% of subtotal";
+        const metadata = getMetadata(store.getState())
+
+        const query: StructuredQuery = Question.create({ databaseId: DATABASE_ID, tableId: ORDERS_TABLE_ID, metadata })
+            .query()
+            .updateExpression(expressionName, ["*", ["field-id", 4], 0.7])
+
+        // Use the count aggregation as an example case (this is equally valid for filters and groupings)
+        const fieldOptions = query.aggregationFieldOptions("sum");
+        const component = mount(getFieldList(query, fieldOptions));
+
+        expect(component.find(`.List-item-title[children="${expressionName}"]`).length).toBe(1);
+    });
+
+    it("should show the query definition tooltip correctly for a segment", async () => {
+        // TODO Atte Keinänen 6/27/17: Check why the result is wrapped in a promise that needs to be resolved manually
+        const segment = await (await createSegment(orders_past_30_days_segment)).payload;
+
+        const store = await createTestStore()
+        await store.dispatch(fetchDatabases());
+        await store.dispatch(fetchTableMetadata(ORDERS_TABLE_ID));
+        await store.dispatch(fetchSegments());
+        const metadata = getMetadata(store.getState())
+
+        const query: StructuredQuery = Question.create({ databaseId: DATABASE_ID, tableId: ORDERS_TABLE_ID, metadata }).query();
+        const component = mount(getFieldList(query, query.filterFieldOptions(), query.filterSegmentOptions()));
+
+        // TODO: This is pretty awkward – each list item could have its own React component for easier traversal
+        // Maybe also TestTooltip should provide an interface (like `tooltipWrapper.instance().show()`) for toggling it?
+        const tooltipTarget = component.find(`.List-item-title[children="${segment.name}"]`)
+            .first()
+            .closest('li')
+            .find(".QuestionTooltipTarget")
+            .parent();
+
+        tooltipTarget.simulate("mouseenter");
+        
+        const tooltipContent = tooltipTarget.closest(TestTooltip).find(TestTooltipContent);
+        expect(tooltipContent.length).toBe(1)
+
+        // eslint-disable-next-line no-irregular-whitespace
+        expect(tooltipContent.find(FilterWidget).last().text()).toMatch(/Created At -30day/);
+    })
+});
\ No newline at end of file
diff --git a/frontend/src/metabase/query_builder/components/FieldList.jsx b/frontend/src/metabase/query_builder/components/FieldList.jsx
index 35f983047a0fbca76f6951026e71bd29332c9338..857cc7ae52ac6167451cb3a5674c36ed4abdafad 100644
--- a/frontend/src/metabase/query_builder/components/FieldList.jsx
+++ b/frontend/src/metabase/query_builder/components/FieldList.jsx
@@ -4,7 +4,7 @@ import React, { Component } from "react";
 
 import AccordianList from "metabase/components/AccordianList.jsx";
 import Icon from "metabase/components/Icon.jsx";
-import Tooltip from "metabase/components/Tooltip.jsx";
+import Tooltip from "metabase/components/Tooltip";
 import PopoverWithTrigger from "metabase/components/PopoverWithTrigger.jsx";
 import QueryDefinitionTooltip from "./QueryDefinitionTooltip.jsx";
 
diff --git a/frontend/src/metabase/query_builder/components/FieldList.spec.jsx b/frontend/src/metabase/query_builder/components/FieldList.spec.jsx
deleted file mode 100644
index 9960b58e7d97689e66a7511a1dcfcf119bd0798d..0000000000000000000000000000000000000000
--- a/frontend/src/metabase/query_builder/components/FieldList.spec.jsx
+++ /dev/null
@@ -1,35 +0,0 @@
-import React from 'react'
-import { mount } from 'enzyme'
-
-import FieldList from './FieldList';
-import Question from "metabase-lib/lib/Question";
-import {
-    DATABASE_ID,
-    ORDERS_TABLE_ID,
-    metadata
-} from "metabase/__support__/sample_dataset_fixture";
-
-import StructuredQuery from "metabase-lib/lib/queries/StructuredQuery";
-
-const getFieldList = (query, fieldOptions) =>
-    <FieldList
-        tableMetadata={query.tableMetadata()}
-        fieldOptions={fieldOptions}
-        customFieldOptions={query.expressions()}
-        onFieldChange={() => {}}
-        enableSubDimensions={false}
-    />
-
-describe('FieldList', () => {
-    it("should allow using expression as aggregation dimension", () => {
-        const expressionName = "70% of subtotal";
-        const query: StructuredQuery = Question.create({databaseId: DATABASE_ID, tableId: ORDERS_TABLE_ID, metadata})
-            .query()
-            .updateExpression(expressionName, ["*", ["field-id", 4], 0.7])
-
-        // Use the count aggregation as an example case (this is equally valid for filters and groupings)
-        const fieldOptions = query.aggregationFieldOptions("sum");
-        const component = mount(getFieldList(query, fieldOptions));
-        expect(component.find(`.List-item-title[children="${expressionName}"]`).length).toBe(1);
-    });
-});
\ No newline at end of file
diff --git a/frontend/src/metabase/query_builder/components/FieldName.jsx b/frontend/src/metabase/query_builder/components/FieldName.jsx
index f9887affef05735b8ffa7b87e63a28d8599276cf..52cfac407576f4a6b3ab6a15d9c60b4c676c136a 100644
--- a/frontend/src/metabase/query_builder/components/FieldName.jsx
+++ b/frontend/src/metabase/query_builder/components/FieldName.jsx
@@ -7,6 +7,7 @@ import Query from "metabase/lib/query";
 
 import Dimension from "metabase-lib/lib/Dimension";
 
+import _ from "underscore";
 import cx from "classnames";
 
 export default class FieldName extends Component {
@@ -21,6 +22,13 @@ export default class FieldName extends Component {
         className: ""
     };
 
+    displayNameForFieldLiteral(tableMetadata, fieldLiteral) {
+        // see if we can find an entry in the table metadata that matches the field literal
+        let matchingField = _.find(tableMetadata.fields, (field) => Query.isFieldLiteral(field.id) && field.id[1] === fieldLiteral[1]); // check whether names of field literals match
+
+        return (matchingField && matchingField.display_name) || fieldLiteral[1];
+    }
+
     render() {
         let { field, tableMetadata, className } = this.props;
 
@@ -30,6 +38,16 @@ export default class FieldName extends Component {
             const dimension = Dimension.parseMBQL(field, tableMetadata && tableMetadata.metadata);
             if (dimension) {
                 parts = dimension.render();
+            }
+            // TODO Atte Keinänen 6/23/17: Move nested queries logic to Dimension subclasses
+            // if the Field in question is a field literal, e.g. ["field-literal", <name>, <type>] just use name as-is
+            else if (Query.isFieldLiteral(field)) {
+                parts.push(<span key="field">{this.displayNameForFieldLiteral(tableMetadata, field)}</span>);
+            }
+            // otherwise if for some weird reason we wound up with a Field Literal inside a field ID,
+            // e.g. ["field-id", ["field-literal", <name>, <type>], still just use the name as-is
+            else if (Query.isLocalField(field) && Query.isFieldLiteral(field[1])) {
+                parts.push(<span key="field">{this.displayNameForFieldLiteral(tableMetadata, field[1])}</span>);
             } else {
                 parts.push(<span key="field">Unknown Field</span>);
             }
diff --git a/frontend/src/metabase/query_builder/components/NativeQueryEditor.integ.spec.jsx b/frontend/src/metabase/query_builder/components/NativeQueryEditor.integ.spec.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..9f34d328d9f63a94442591841ee752c9b73638aa
--- /dev/null
+++ b/frontend/src/metabase/query_builder/components/NativeQueryEditor.integ.spec.jsx
@@ -0,0 +1,5 @@
+describe("NativeQueryEditor", () => {
+    it("lets you create a SQL question with a field filter variable", () => {
+        pending();
+    })
+})
\ No newline at end of file
diff --git a/frontend/src/metabase/query_builder/components/QueryHeader.jsx b/frontend/src/metabase/query_builder/components/QueryHeader.jsx
index 2c64dbea8040275560954994be4bf4c43e2fc528..ad62ff03d5edbda925383e8fef1a3861542a8229 100644
--- a/frontend/src/metabase/query_builder/components/QueryHeader.jsx
+++ b/frontend/src/metabase/query_builder/components/QueryHeader.jsx
@@ -1,6 +1,7 @@
 import React, { Component } from "react";
 import PropTypes from "prop-types";
 import { Link } from "react-router";
+import { connect } from "react-redux";
 
 import QueryModeButton from "./QueryModeButton.jsx";
 
@@ -19,6 +20,8 @@ import ArchiveQuestionModal from "metabase/query_builder/containers/ArchiveQuest
 
 import SaveQuestionModal from 'metabase/containers/SaveQuestionModal.jsx';
 
+import { clearRequestState } from "metabase/redux/requests";
+
 import { CardApi, RevisionApi } from "metabase/services";
 
 import MetabaseAnalytics from "metabase/lib/analytics";
@@ -31,7 +34,11 @@ import _ from "underscore";
 import NativeQuery from "metabase-lib/lib/queries/NativeQuery";
 import Utils from "metabase/lib/utils";
 
+const mapDispatchToProps = {
+    clearRequestState
+};
 
+@connect(null, mapDispatchToProps)
 export default class QueryHeader extends Component {
     constructor(props, context) {
         super(props, context);
@@ -92,6 +99,25 @@ export default class QueryHeader extends Component {
             return card
         }
     }
+
+    /// Add result_metadata and metadata_checksum columns to card as expected by the endpoints used for saving
+    /// and updating Cards. These values are returned as part of Query Processor results and fetched from there
+    addResultMetadata(card) {
+        let metadata = this.props.result && this.props.result.data && this.props.result.data.results_metadata;
+        let metadataChecksum = metadata && metadata.checksum;
+        let metadataColumns = metadata && metadata.columns;
+
+        card.result_metadata = metadataColumns;
+        card.metadata_checksum = metadataChecksum;
+    }
+
+    /// remove the databases in the store that are used to populate the QB databases list.
+    /// This is done when saving a Card because the newly saved card will be elligable for use as a source query
+    /// so we want the databases list to be re-fetched next time we hit "New Question" so it shows up
+    clearQBDatabases() {
+        this.props.clearRequestState({ statePath: ["metadata", "databases"] });
+    }
+
     onCreate(card, addToDash) {
         // MBQL->NATIVE
         // if we are a native query with an MBQL query definition, remove the old MBQL stuff (happens when going from mbql -> native)
@@ -102,9 +128,13 @@ export default class QueryHeader extends Component {
         // }
 
         const cleanedCard = this._getCleanedCard(card);
+        this.addResultMetadata(cleanedCard);
+
         // TODO: reduxify
         this.requestPromise = cancelable(CardApi.create(cleanedCard));
         return this.requestPromise.then(newCard => {
+            this.clearQBDatabases();
+
             this.props.notifyCardCreatedFn(newCard);
 
             this.setState({
@@ -124,9 +154,13 @@ export default class QueryHeader extends Component {
         // }
 
         const cleanedCard = this._getCleanedCard(card);
+        this.addResultMetadata(cleanedCard);
+
         // TODO: reduxify
         this.requestPromise = cancelable(CardApi.update(cleanedCard));
         return this.requestPromise.then(updatedCard => {
+            this.clearQBDatabases();
+
             if (this.props.fromUrl) {
                 this.onGoBack();
                 return;
diff --git a/frontend/src/metabase/query_builder/components/QueryVisualization.jsx b/frontend/src/metabase/query_builder/components/QueryVisualization.jsx
index 546af0bff19d1f94b3602e5376fceed6d9f3a910..ef5edfc86d474da4b684c1dc83b1dac128f70826 100644
--- a/frontend/src/metabase/query_builder/components/QueryVisualization.jsx
+++ b/frontend/src/metabase/query_builder/components/QueryVisualization.jsx
@@ -242,7 +242,7 @@ export default class QueryVisualization extends Component {
     }
 }
 
-const VisualizationEmptyState = ({showTutorialLink}) =>
+export const VisualizationEmptyState = ({showTutorialLink}) =>
     <div className="flex full layout-centered text-grey-1 flex-column">
         <h1>If you give me some data I can show you something cool. Run a Query!</h1>
         { showTutorialLink && <Link to={Urls.question(null, "?tutorial")} className="link cursor-pointer my2">How do I use this thing?</Link> }
diff --git a/frontend/src/metabase/query_builder/components/VisualizationError.jsx b/frontend/src/metabase/query_builder/components/VisualizationError.jsx
index 80b795c90fe77b0a274bedb252da92363cd94ca8..587ca14eee21cdbf8e8e831d1d9551b09d29c230 100644
--- a/frontend/src/metabase/query_builder/components/VisualizationError.jsx
+++ b/frontend/src/metabase/query_builder/components/VisualizationError.jsx
@@ -33,7 +33,8 @@ class VisualizationError extends Component {
 
   render () {
       const { card, duration, error } = this.props
-      if (typeof error.status === "number") {
+
+      if (error && typeof error.status === "number") {
           // Assume if the request took more than 15 seconds it was due to a timeout
           // Some platforms like Heroku return a 503 for numerous types of errors so we can't use the status code to distinguish between timeouts and other failures.
           if (duration > 15*1000) {
@@ -51,7 +52,7 @@ class VisualizationError extends Component {
                         action={<EmailAdmin />}
                     />
           }
-      } else if (card.dataset_query && card.dataset_query.type === 'native') {
+      } else if (card && card.dataset_query && card.dataset_query.type === 'native') {
           // always show errors for native queries
           return (
               <div className="QueryError flex full align-center text-error">
@@ -66,7 +67,7 @@ class VisualizationError extends Component {
       } else {
           return (
               <div className="QueryError2 flex full justify-center">
-                  <div className="QueryError-image QueryError-image--queryError mr4"></div>
+                  <div className="QueryError-image QueryError-image--queryError mr4" />
                   <div className="QueryError2-details">
                       <h1 className="text-bold">There was a problem with your question</h1>
                       <p className="QueryError-messageText">Most of the time this is caused by an invalid selection or bad input value.  Double check your inputs and retry your query.</p>
diff --git a/frontend/src/metabase/query_builder/components/dataref/DetailPane.jsx b/frontend/src/metabase/query_builder/components/dataref/DetailPane.jsx
index cd8203745c1096b91d0e0537ad6453e916822f29..6ebc166666bf52eb0fb1ba1d885313227a8c587a 100644
--- a/frontend/src/metabase/query_builder/components/dataref/DetailPane.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/DetailPane.jsx
@@ -4,7 +4,7 @@ import PropTypes from "prop-types";
 
 import cx from "classnames";
 
-const DetailPane = ({ name, description, error, usefulQuestions, useForCurrentQuestion, extra }) =>
+const DetailPane = ({ name, description, usefulQuestions, useForCurrentQuestion, extra }) =>
     <div>
         <h1>{name}</h1>
         <p className={cx({ "text-grey-3": !description })}>
@@ -35,7 +35,6 @@ const DetailPane = ({ name, description, error, usefulQuestions, useForCurrentQu
             </div>
         : null }
         {extra}
-        <div>{error}</div>
     </div>
 
 DetailPane.propTypes = {
diff --git a/frontend/src/metabase/query_builder/components/dataref/FieldPane.integ.spec.js b/frontend/src/metabase/query_builder/components/dataref/FieldPane.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..ea6024926baed3d1daabefad3deafdb806bbbc90
--- /dev/null
+++ b/frontend/src/metabase/query_builder/components/dataref/FieldPane.integ.spec.js
@@ -0,0 +1,90 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from "enzyme";
+
+import {
+    INITIALIZE_QB, QUERY_COMPLETED, setQuerySourceTable,
+    TOGGLE_DATA_REFERENCE
+} from "metabase/query_builder/actions";
+import { delay } from "metabase/lib/promise"
+
+import QueryBuilder from "metabase/query_builder/containers/QueryBuilder";
+import DataReference from "metabase/query_builder/components/dataref/DataReference";
+import { FETCH_TABLE_METADATA } from "metabase/redux/metadata";
+import QueryButton from "metabase/components/QueryButton";
+import Table from "metabase/visualizations/visualizations/Table";
+import UseForButton from "metabase/query_builder/components/dataref/UseForButton";
+
+// Currently a lot of duplication with FieldPane tests
+describe("FieldPane", () => {
+    let store = null;
+    let queryBuilder = null;
+
+    beforeAll(async () => {
+        await login();
+        store = await createTestStore()
+
+        store.pushPath("/question");
+        queryBuilder = mount(store.connectContainer(<QueryBuilder />));
+        await store.waitForActions([INITIALIZE_QB]);
+    })
+
+    // NOTE: These test cases are intentionally stateful
+    // (doing the whole app rendering thing in every single test case would probably slow things down)
+
+    it("opens properly from QB", async () => {
+        // open data reference sidebar by clicking button
+        queryBuilder.find(".Icon-reference").simulate("click");
+        await store.waitForActions([TOGGLE_DATA_REFERENCE]);
+
+        const dataReference = queryBuilder.find(DataReference);
+        expect(dataReference.length).toBe(1);
+
+        dataReference.find('a[children="Orders"]').simulate("click");
+
+        // TODO: Refactor TablePane so that it uses redux/metadata actions instead of doing inlined API calls
+        // then we can replace this with `store.waitForActions([FETCH_TABLE_FOREIGN_KEYS])` or similar
+        await delay(3000)
+
+        dataReference.find(`a[children="Created At"]`).first().simulate("click")
+
+        await store.waitForActions([FETCH_TABLE_METADATA]);
+    });
+
+    it("lets you group by Created At", async () => {
+        const getUseForButton = () => queryBuilder.find(DataReference).find(UseForButton);
+
+        expect(getUseForButton().length).toBe(0);
+
+        await store.dispatch(setQuerySourceTable(1))
+        // eslint-disable-line react/no-irregular-whitespace
+        expect(getUseForButton().text()).toMatch(/Group by/);
+
+        getUseForButton().simulate('click');
+        await store.waitForActions([QUERY_COMPLETED]);
+        store.resetDispatchedActions()
+
+        // after the breakout has been applied, the button shouldn't be visible anymore
+        expect(getUseForButton().length).toBe(0);
+    })
+
+    it("lets you see all distinct values of Created At", async () => {
+        const distinctValuesButton = queryBuilder.find(DataReference).find(QueryButton).at(0);
+
+        try {
+            distinctValuesButton.children().first().simulate("click");
+        } catch(e) {
+            // QueryButton uses react-router Link which always throws an error if it's called without a parent Router object
+            // Now we are just using the onClick handler of Link so we don't have to care about that
+        }
+
+        await store.waitForActions([QUERY_COMPLETED]);
+        store.resetDispatchedActions()
+
+        expect(queryBuilder.find(Table).length).toBe(1)
+    });
+});
diff --git a/frontend/src/metabase/query_builder/components/dataref/FieldPane.jsx b/frontend/src/metabase/query_builder/components/dataref/FieldPane.jsx
index 4e8141f50f3f506f300d0bec8ae1f624fdbc16da..f8b045b0f5aceb165459f11b072aca9424ec3368 100644
--- a/frontend/src/metabase/query_builder/components/dataref/FieldPane.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/FieldPane.jsx
@@ -6,23 +6,30 @@ import DetailPane from "./DetailPane.jsx";
 import QueryButton from "metabase/components/QueryButton.jsx";
 import UseForButton from "./UseForButton.jsx";
 
+import { fetchTableMetadata } from "metabase/redux/metadata";
+import { getMetadata } from "metabase/selectors/metadata";
 import { createCard } from "metabase/lib/card";
 import Query, { createQuery } from "metabase/lib/query";
 import { isDimension, isSummable } from "metabase/lib/schema_metadata";
 import inflection from 'inflection';
 
 import _ from "underscore";
+import StructuredQuery from "metabase-lib/lib/queries/StructuredQuery";
+import { connect } from "react-redux";
+import Dimension from "metabase-lib/lib/Dimension";
 
+const mapDispatchToProps = {
+    fetchTableMetadata,
+};
+
+const mapStateToProps = (state, props) => ({
+    metadata: getMetadata(state, props)
+})
+
+@connect(mapStateToProps, mapDispatchToProps)
 export default class FieldPane extends Component {
     constructor(props, context) {
         super(props, context);
-
-        this.state = {
-            table: undefined,
-            tableForeignKeys: undefined
-        };
-
-        _.bindAll(this, "filterBy", "groupBy", "setQuerySum", "setQueryDistinct", "setQueryCountGroupedBy");
     }
 
     static propTypes = {
@@ -30,96 +37,116 @@ export default class FieldPane extends Component {
         datasetQuery: PropTypes.object,
         question: PropTypes.object,
         originalQuestion: PropTypes.object,
-        loadTableAndForeignKeysFn: PropTypes.func.isRequired,
+        metadata: PropTypes.object,
+        fetchTableMetadata: PropTypes.func.isRequired,
         runQuestionQuery: PropTypes.func.isRequired,
         setDatasetQuery: PropTypes.func.isRequired,
-        setCardAndRun: PropTypes.func.isRequired
+        setCardAndRun: PropTypes.func.isRequired,
+        updateQuestion: PropTypes.func.isRequired
     };
 
     componentWillMount() {
-        this.props.loadTableAndForeignKeysFn(this.props.field.table_id).then((result) => {
-            this.setState({
-                table: result.table,
-                tableForeignKeys: result.foreignKeys
-            });
-        }).catch((error) => {
-            this.setState({
-                error: "An error occurred loading the table"
-            });
-        });
+        this.props.fetchTableMetadata(this.props.field.table_id);
     }
 
-    filterBy() {
-        var datasetQuery = this.setDatabaseAndTable();
-        // Add an aggregation so both aggregation and filter popovers aren't visible
-        if (!Query.hasValidAggregation(datasetQuery.query)) {
-            Query.clearAggregations(datasetQuery.query);
-        }
-        Query.addFilter(datasetQuery.query, [null, this.props.field.id, null]);
-        this.props.setDatasetQuery(datasetQuery);
-    }
+    // See the note in render() method about filterBy
+    // filterBy() {
+    //     var datasetQuery = this.setDatabaseAndTable();
+    //     // Add an aggregation so both aggregation and filter popovers aren't visible
+    //     if (!Query.hasValidAggregation(datasetQuery.query)) {
+    //         Query.clearAggregations(datasetQuery.query);
+    //     }
+    //     Query.addFilter(datasetQuery.query, [null, this.props.field.id, null]);
+    //     this.props.setDatasetQuery(datasetQuery);
+    // }
+
+    groupBy = () => {
+
+        let { question, metadata, field } = this.props;
+        let query = question.query();
+
+        if (query instanceof StructuredQuery) {
+            // Add an aggregation so both aggregation and filter popovers aren't visible
+            if (!Query.hasValidAggregation(query.datasetQuery().query)) {
+                query = query.clearAggregations()
+            }
 
-    groupBy() {
-        let { datasetQuery, question } = this.props;
-        if (!Query.hasValidAggregation(datasetQuery.query)) {
-            Query.clearAggregations(datasetQuery.query);
+            const defaultBreakout = metadata.fields[field.id].getDefaultBreakout();
+            query = query.addBreakout(defaultBreakout);
+
+            this.props.updateQuestion(query.question())
+            this.props.runQuestionQuery();
         }
-        Query.addBreakout(datasetQuery.query, this.props.field.id);
-        this.props.setDatasetQuery(datasetQuery);
-        this.props.runQuestionQuery({ overrideWithCard: question.card() });
     }
 
-    newCard() {
+    newCard = () => {
+        const { metadata, field } = this.props;
+        const tableId = field.table_id;
+        const dbId = metadata.tables[tableId].database.id;
+
         let card = createCard();
-        card.dataset_query = createQuery("query", this.state.table.db_id, this.state.table.id);
+        card.dataset_query = createQuery("query", dbId, tableId);
         return card;
     }
 
-    setQuerySum() {
+    setQuerySum = () => {
         let card = this.newCard();
         card.dataset_query.query.aggregation = ["sum", this.props.field.id];
         this.props.setCardAndRun(card);
     }
 
-    setQueryDistinct() {
+    setQueryDistinct = () => {
+        const { metadata, field } = this.props;
+        const defaultBreakout = metadata.fields[field.id].getDefaultBreakout();
+
         let card = this.newCard();
         card.dataset_query.query.aggregation = ["rows"];
-        card.dataset_query.query.breakout = [this.props.field.id];
+        card.dataset_query.query.breakout = [defaultBreakout];
         this.props.setCardAndRun(card);
     }
 
-    setQueryCountGroupedBy(chartType) {
+    setQueryCountGroupedBy = (chartType) => {
+        const { metadata, field } = this.props;
+        const defaultBreakout = metadata.fields[field.id].getDefaultBreakout();
+
         let card = this.newCard();
         card.dataset_query.query.aggregation = ["count"];
-        card.dataset_query.query.breakout = [this.props.field.id];
+        card.dataset_query.query.breakout = [defaultBreakout];
         card.display = chartType;
         this.props.setCardAndRun(card);
     }
 
+    isBreakoutWithCurrentField = (breakout) => {
+        const { field, metadata } = this.props;
+        const dimension = Dimension.parseMBQL(breakout, metadata);
+        return dimension && dimension.field().id === field.id;
+    }
+
     render() {
-        let { field, datasetQuery } = this.props;
-        let { table, error } = this.state;
+        let { field, question } = this.props;
+
+        const query = question.query();
 
         let fieldName = field.display_name;
-        let tableName = table ? table.display_name : "";
+        let tableName = query.table() ? query.table().display_name : "";
 
         let useForCurrentQuestion = [],
             usefulQuestions = [];
 
         // determine if the selected field is a valid dimension on this table
         let validBreakout = false;
-        if (this.state.table) {
-            const validDimensions = _.filter(table.fields, isDimension);
+        if (query.table()) {
+            const validDimensions = _.filter(query.table().fields, isDimension);
             validBreakout = _.some(validDimensions, f => f.id === field.id);
         }
 
         // TODO: allow for filters/grouping via foreign keys
-        if (!datasetQuery.query || datasetQuery.query.source_table == undefined || datasetQuery.query.source_table === field.table_id) {
+        if (query instanceof StructuredQuery && query.tableId() === field.table_id) {
             // NOTE: disabled this for now because we need a way to capture the completed filter before adding it to the query, or to pop open the filter widget here?
             // useForCurrentQuestion.push(<UseForButton title={"Filter by " + name} onClick={this.filterBy} />);
 
             // current field must be a valid breakout option for this table AND cannot already be in the breakout clause of our query
-            if (validBreakout && this.state.table.id === datasetQuery.query.source_table && (datasetQuery.query.breakout && !_.contains(datasetQuery.query.breakout, field.id))) {
+            if (validBreakout && !_.some(query.breakouts(), this.isBreakoutWithCurrentField)) {
                 useForCurrentQuestion.push(<UseForButton title={"Group by " + name} onClick={this.groupBy} />);
             }
         }
@@ -140,7 +167,6 @@ export default class FieldPane extends Component {
                 description={field.description}
                 useForCurrentQuestion={useForCurrentQuestion}
                 usefulQuestions={usefulQuestions}
-                error={error}
             />
         );
     }
diff --git a/frontend/src/metabase/query_builder/components/dataref/MetricPane.integ.spec.js b/frontend/src/metabase/query_builder/components/dataref/MetricPane.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..5e8f00e3caf16c754947e3c9d27e5b6815f310a4
--- /dev/null
+++ b/frontend/src/metabase/query_builder/components/dataref/MetricPane.integ.spec.js
@@ -0,0 +1,77 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from "enzyme";
+
+import { INITIALIZE_QB, QUERY_COMPLETED, TOGGLE_DATA_REFERENCE } from "metabase/query_builder/actions";
+import { delay } from "metabase/lib/promise"
+
+import QueryBuilder from "metabase/query_builder/containers/QueryBuilder";
+import DataReference from "metabase/query_builder/components/dataref/DataReference";
+import { vendor_count_metric } from "metabase/__support__/sample_dataset_fixture";
+import { createMetric } from "metabase/admin/datamodel/datamodel";
+import { FETCH_TABLE_METADATA } from "metabase/redux/metadata";
+import QueryDefinition from "metabase/query_builder/components/dataref/QueryDefinition";
+import QueryButton from "metabase/components/QueryButton";
+import Scalar from "metabase/visualizations/visualizations/Scalar";
+
+describe("MetricPane", () => {
+    let store = null;
+    let queryBuilder = null;
+
+    beforeAll(async () => {
+        await login();
+        await createMetric(vendor_count_metric);
+        store = await createTestStore()
+
+        store.pushPath("/question");
+        queryBuilder = mount(store.connectContainer(<QueryBuilder />));
+        await store.waitForActions([INITIALIZE_QB]);
+    })
+
+    // NOTE: These test cases are intentionally stateful
+    // (doing the whole app rendering thing in every single test case would probably slow things down)
+
+    it("opens properly from QB", async () => {
+        // open data reference sidebar by clicking button
+        queryBuilder.find(".Icon-reference").simulate("click");
+        await store.waitForActions([TOGGLE_DATA_REFERENCE]);
+
+        const dataReference = queryBuilder.find(DataReference);
+        expect(dataReference.length).toBe(1);
+
+        dataReference.find('a[children="Products"]').simulate("click");
+
+        // TODO: Refactor TablePane so that it uses redux/metadata actions instead of doing inlined API calls
+        // then we can replace this with `store.waitForActions([FETCH_TABLE_FOREIGN_KEYS])` or similar
+        await delay(3000)
+
+        store.resetDispatchedActions() // make sure that we wait for the newest actions
+        dataReference.find(`a[children="${vendor_count_metric.name}"]`).first().simulate("click")
+
+        await store.waitForActions([FETCH_TABLE_METADATA]);
+    });
+
+    it("shows you the correct definition", () => {
+        const queryDefinition = queryBuilder.find(DataReference).find(QueryDefinition);
+        expect(queryDefinition.text()).toMatch(/Number of distinct valuesofVendor/);
+    })
+
+    it("lets you see the vendor count", async () => {
+        const queryButton = queryBuilder.find(DataReference).find(QueryButton);
+
+        try {
+            queryButton.children().first().simulate("click");
+        } catch(e) {
+            // QueryButton uses react-router Link which always throws an error if it's called without a parent Router object
+            // Now we are just using the onClick handler of Link so we don't have to care about that
+        }
+
+        await store.waitForActions([QUERY_COMPLETED]);
+
+        expect(queryBuilder.find(Scalar).text()).toBe("200")
+    });
+});
diff --git a/frontend/src/metabase/query_builder/components/dataref/MetricPane.jsx b/frontend/src/metabase/query_builder/components/dataref/MetricPane.jsx
index 97ef33458abf8e696bd3e9911f99dbbb04f176ca..8c90c9edf7b790b4c7a874e60635ab4577ee3479 100644
--- a/frontend/src/metabase/query_builder/components/dataref/MetricPane.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/MetricPane.jsx
@@ -1,5 +1,6 @@
 /* eslint "react/prop-types": "warn" */
 import React, { Component } from "react";
+import { connect } from "react-redux";
 import PropTypes from "prop-types";
 
 import DetailPane from "./DetailPane.jsx";
@@ -10,45 +11,51 @@ import { createCard } from "metabase/lib/card";
 import { createQuery } from "metabase/lib/query";
 
 import _ from "underscore";
+import { fetchTableMetadata } from "metabase/redux/metadata";
 
+import { getMetadata } from "metabase/selectors/metadata";
+
+const mapDispatchToProps = {
+    fetchTableMetadata,
+};
+
+const mapStateToProps = (state, props) => ({
+    metadata: getMetadata(state, props)
+})
+
+@connect(mapStateToProps, mapDispatchToProps)
 export default class MetricPane extends Component {
     constructor(props, context) {
         super(props, context);
 
-        this.state = {
-            table: undefined,
-            tableForeignKeys: undefined
-        };
-
         _.bindAll(this, "setQueryMetric");
     }
 
     static propTypes = {
         metric: PropTypes.object.isRequired,
         query: PropTypes.object,
-        loadTableAndForeignKeysFn: PropTypes.func.isRequired,
+        fetchTableMetadata: PropTypes.func.isRequired,
         runQuestionQuery: PropTypes.func.isRequired,
         setDatasetQuery: PropTypes.func.isRequired,
-        setCardAndRun: PropTypes.func.isRequired
+        setCardAndRun: PropTypes.func.isRequired,
+        metadata: PropTypes.object
     };
 
     componentWillMount() {
-        this.props.loadTableAndForeignKeysFn(this.props.metric.table_id).then((result) => {
-            this.setState({
-                table: result.table,
-                tableForeignKeys: result.foreignKeys
-            });
-        }).catch((error) => {
-            this.setState({
-                error: "An error occurred loading the table"
-            });
-        });
+        this.props.fetchTableMetadata(this.props.metric.table_id);
     }
 
     newCard() {
-        let card = createCard();
-        card.dataset_query = createQuery("query", this.state.table.db_id, this.state.table.id);
-        return card;
+        const { metric, metadata } = this.props;
+        const table = metadata && metadata.tables[metric.table_id];
+
+        if (table) {
+            let card = createCard();
+            card.dataset_query = createQuery("query", table.db_id, table.id);
+            return card;
+        } else {
+            throw new Error("Could not find the table metadata prior to creating a new question")
+        }
     }
 
     setQueryMetric() {
@@ -58,8 +65,7 @@ export default class MetricPane extends Component {
     }
 
     render() {
-        let { metric } = this.props;
-        let { table, error } = this.state;
+        let { metric, metadata } = this.props;
 
         let metricName = metric.name;
 
@@ -74,11 +80,10 @@ export default class MetricPane extends Component {
                 description={metric.description}
                 useForCurrentQuestion={useForCurrentQuestion}
                 usefulQuestions={usefulQuestions}
-                error={error}
-                extra={table &&
+                extra={metadata &&
                     <div>
                         <p className="text-bold">Metric Definition</p>
-                        <QueryDefinition object={metric} tableMetadata={table} />
+                        <QueryDefinition object={metric} tableMetadata={metadata.tables[metric.table_id]} />
                     </div>
                 }
             />
diff --git a/frontend/src/metabase/query_builder/components/dataref/QueryDefinition.jsx b/frontend/src/metabase/query_builder/components/dataref/QueryDefinition.jsx
index 3cc284aa23362feba9ac66eeb4f56a16c261c9af..00e289dda749f79720311dba93aa9f63619e73ac 100644
--- a/frontend/src/metabase/query_builder/components/dataref/QueryDefinition.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/QueryDefinition.jsx
@@ -11,7 +11,7 @@ const QueryDefinition = ({ className, object, tableMetadata }) => {
         <div className={className} style={{ pointerEvents: "none" }}>
             { object.definition.aggregation &&
                 <AggregationWidget
-                    aggregation={object.definition.aggregation}
+                    aggregation={object.definition.aggregation[0]}
                     tableMetadata={tableMetadata}
                 />
             }
diff --git a/frontend/src/metabase/query_builder/components/dataref/SegmentPane.integ.spec.js b/frontend/src/metabase/query_builder/components/dataref/SegmentPane.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..90601a81d81b26da3527087029cb15ad77cd2c88
--- /dev/null
+++ b/frontend/src/metabase/query_builder/components/dataref/SegmentPane.integ.spec.js
@@ -0,0 +1,124 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from "enzyme";
+
+import {
+    INITIALIZE_QB, LOAD_TABLE_METADATA, QUERY_COMPLETED, setQuerySourceTable,
+    TOGGLE_DATA_REFERENCE
+} from "metabase/query_builder/actions";
+import { delay } from "metabase/lib/promise"
+
+import QueryBuilder from "metabase/query_builder/containers/QueryBuilder";
+import DataReference from "metabase/query_builder/components/dataref/DataReference";
+import { orders_past_30_days_segment } from "metabase/__support__/sample_dataset_fixture";
+import { FETCH_TABLE_METADATA } from "metabase/redux/metadata";
+import QueryDefinition from "metabase/query_builder/components/dataref/QueryDefinition";
+import QueryButton from "metabase/components/QueryButton";
+import Scalar from "metabase/visualizations/visualizations/Scalar";
+import Table from "metabase/visualizations/visualizations/Table";
+import UseForButton from "metabase/query_builder/components/dataref/UseForButton";
+import { SegmentApi } from "metabase/services";
+
+// Currently a lot of duplication with SegmentPane tests
+describe("SegmentPane", () => {
+    let store = null;
+    let queryBuilder = null;
+    let segment = null;
+
+    beforeAll(async () => {
+        await login();
+        segment = await SegmentApi.create(orders_past_30_days_segment);
+        store = await createTestStore()
+
+        store.pushPath("/question");
+        queryBuilder = mount(store.connectContainer(<QueryBuilder />));
+        await store.waitForActions([INITIALIZE_QB]);
+    })
+
+    afterAll(async() => {
+        await SegmentApi.delete({
+            segmentId: segment.id,
+            revision_message: "Please"
+        });
+    })
+
+    // NOTE: These test cases are intentionally stateful
+    // (doing the whole app rendering thing in every single test case would probably slow things down)
+
+    it("opens properly from QB", async () => {
+        // open data reference sidebar by clicking button
+        queryBuilder.find(".Icon-reference").simulate("click");
+        await store.waitForActions([TOGGLE_DATA_REFERENCE]);
+
+        const dataReference = queryBuilder.find(DataReference);
+        expect(dataReference.length).toBe(1);
+
+        dataReference.find('a[children="Orders"]').simulate("click");
+
+        // TODO: Refactor TablePane so that it uses redux/metadata actions instead of doing inlined API calls
+        // then we can replace this with `store.waitForActions([FETCH_TABLE_FOREIGN_KEYS])` or similar
+        await delay(3000)
+
+        store.resetDispatchedActions() // make sure that we wait for the newest actions
+        dataReference.find(`a[children="${orders_past_30_days_segment.name}"]`).first().simulate("click")
+
+        await store.waitForActions([FETCH_TABLE_METADATA]);
+    });
+
+    it("shows you the correct segment definition", () => {
+        const queryDefinition = queryBuilder.find(DataReference).find(QueryDefinition);
+        // eslint-disable-next-line no-irregular-whitespace
+        expect(queryDefinition.text()).toMatch(/Created At -30day/);
+    })
+
+    it("lets you apply the filter to your current query", async () => {
+        await store.dispatch(setQuerySourceTable(1))
+        await store.waitForActions(LOAD_TABLE_METADATA);
+
+        const filterByButton = queryBuilder.find(DataReference).find(UseForButton).first();
+        filterByButton.children().first().simulate("click");
+
+        await store.waitForActions([QUERY_COMPLETED]);
+        store.resetDispatchedActions()
+
+        expect(queryBuilder.find(DataReference).find(UseForButton).length).toBe(0);
+    });
+
+    it("lets you see count of rows for past 30 days", async () => {
+        const numberQueryButton = queryBuilder.find(DataReference).find(QueryButton).at(0);
+
+        try {
+            numberQueryButton.children().first().simulate("click");
+        } catch(e) {
+            // QueryButton uses react-router Link which always throws an error if it's called without a parent Router object
+            // Now we are just using the onClick handler of Link so we don't have to care about that
+        }
+
+        await store.waitForActions([QUERY_COMPLETED]);
+        store.resetDispatchedActions()
+
+        // The value changes daily which wasn't originally taken into account
+        // expect(queryBuilder.find(Scalar).text()).toBe("1,236")
+        expect(queryBuilder.find(Scalar).text().length).toBe(5)
+    });
+
+    it("lets you see raw data for past 30 days", async () => {
+        const allQueryButton = queryBuilder.find(DataReference).find(QueryButton).at(1);
+
+        try {
+            allQueryButton.children().first().simulate("click");
+        } catch(e) {
+            // QueryButton uses react-router Link which always throws an error if it's called without a parent Router object
+            // Now we are just using the onClick handler of Link so we don't have to care about that
+        }
+
+        await store.waitForActions([QUERY_COMPLETED]);
+        store.resetDispatchedActions()
+
+        expect(queryBuilder.find(Table).length).toBe(1)
+    });
+});
diff --git a/frontend/src/metabase/query_builder/components/dataref/SegmentPane.jsx b/frontend/src/metabase/query_builder/components/dataref/SegmentPane.jsx
index a1747d336a1646c0d5c2664008c2ec5616d1291f..cc2c21af640cf089c7f9d9cd80646a99ce8c8480 100644
--- a/frontend/src/metabase/query_builder/components/dataref/SegmentPane.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/SegmentPane.jsx
@@ -1,6 +1,10 @@
 /* eslint "react/prop-types": "warn" */
 import React, { Component } from "react";
 import PropTypes from "prop-types";
+import { connect } from "react-redux";
+
+import { fetchTableMetadata } from "metabase/redux/metadata";
+import { getMetadata } from "metabase/selectors/metadata";
 
 import DetailPane from "./DetailPane.jsx";
 import QueryButton from "metabase/components/QueryButton.jsx";
@@ -11,60 +15,69 @@ import { createCard } from "metabase/lib/card";
 import Query, { createQuery } from "metabase/lib/query";
 
 import _ from "underscore";
+import StructuredQuery from "metabase-lib/lib/queries/StructuredQuery";
+
+const mapDispatchToProps = {
+    fetchTableMetadata,
+};
+
+const mapStateToProps = (state, props) => ({
+    metadata: getMetadata(state, props)
+})
 
+@connect(mapStateToProps, mapDispatchToProps)
 export default class SegmentPane extends Component {
     constructor(props, context) {
         super(props, context);
 
-        this.state = {
-            table: undefined,
-            tableForeignKeys: undefined
-        };
-
         _.bindAll(this, "filterBy", "setQueryFilteredBy", "setQueryCountFilteredBy");
     }
 
     static propTypes = {
         segment: PropTypes.object.isRequired,
         datasetQuery: PropTypes.object,
-        loadTableAndForeignKeysFn: PropTypes.func.isRequired,
+        fetchTableMetadata: PropTypes.func.isRequired,
         runQuestionQuery: PropTypes.func.isRequired,
-        setDatasetQuery: PropTypes.func.isRequired,
+        updateQuestion: PropTypes.func.isRequired,
         setCardAndRun: PropTypes.func.isRequired,
         question: PropTypes.object.isRequired,
-        originalQuestion: PropTypes.object.isRequired
+        originalQuestion: PropTypes.object.isRequired,
+        metadata: PropTypes.object.isRequired
     };
 
     componentWillMount() {
-        this.props.loadTableAndForeignKeysFn(this.props.segment.table_id).then((result) => {
-            this.setState({
-                table: result.table,
-                tableForeignKeys: result.foreignKeys
-            });
-        }).catch((error) => {
-            this.setState({
-                error: "An error occurred loading the table"
-            });
-        });
+        this.props.fetchTableMetadata(this.props.segment.table_id);
     }
 
     filterBy() {
-        let { datasetQuery } = this.props;
-        // Add an aggregation so both aggregation and filter popovers aren't visible
-        if (!Query.hasValidAggregation(datasetQuery.query)) {
-            Query.clearAggregations(datasetQuery.query);
+        const { question } = this.props;
+        let query = question.query();
+
+        if (query instanceof StructuredQuery) {
+            // Add an aggregation so both aggregation and filter popovers aren't visible
+            if (!Query.hasValidAggregation(query.datasetQuery().query)) {
+                query = query.clearAggregations()
+            }
+
+            query = query.addFilter(["SEGMENT", this.props.segment.id]);
+
+            this.props.updateQuestion(query.question())
+            this.props.runQuestionQuery();
         }
-        Query.addFilter(datasetQuery.query, ["SEGMENT", this.props.segment.id]);
-        this.props.setDatasetQuery(datasetQuery);
-        this.props.runQuestionQuery();
     }
 
     newCard() {
-        let card = createCard();
-        card.dataset_query = createQuery("query", this.state.table.db_id, this.state.table.id);
-        return card;
+        const { segment, metadata } = this.props;
+        const table = metadata && metadata.tables[segment.table_id];
+
+        if (table) {
+            let card = createCard();
+            card.dataset_query = createQuery("query", table.db_id, table.id);
+            return card;
+        } else {
+            throw new Error("Could not find the table metadata prior to creating a new question")
+        }
     }
-
     setQueryFilteredBy() {
         let card = this.newCard();
         card.dataset_query.query.aggregation = ["rows"];
@@ -80,15 +93,19 @@ export default class SegmentPane extends Component {
     }
 
     render() {
-        let { segment, datasetQuery } = this.props;
-        let { error, table } = this.state;
+        let { segment, metadata, question } = this.props;
+        const query = question.query();
 
         let segmentName = segment.name;
 
         let useForCurrentQuestion = [];
         let usefulQuestions = [];
 
-        if (datasetQuery.query && datasetQuery.query.source_table === segment.table_id && !_.findWhere(Query.getFilters(datasetQuery.query), { [0]: "SEGMENT", [1]: segment.id })) {
+
+        if (query instanceof StructuredQuery &&
+            query.tableId() === segment.table_id &&
+            !_.findWhere(query.filters(), {[0]: "SEGMENT", [1]: segment.id})) {
+
             useForCurrentQuestion.push(<UseForButton title={"Filter by " + segmentName} onClick={this.filterBy} />);
         }
 
@@ -101,12 +118,11 @@ export default class SegmentPane extends Component {
                 description={segment.description}
                 useForCurrentQuestion={useForCurrentQuestion}
                 usefulQuestions={usefulQuestions}
-                error={error}
-                extra={table &&
-                    <div>
-                        <p className="text-bold">Segment Definition</p>
-                        <QueryDefinition object={segment} tableMetadata={table} />
-                    </div>
+                extra={metadata &&
+                <div>
+                    <p className="text-bold">Segment Definition</p>
+                    <QueryDefinition object={segment} tableMetadata={metadata.tables[segment.table_id]} />
+                </div>
                 }
             />
         );
diff --git a/frontend/src/metabase/query_builder/components/dataref/TablePane.jsx b/frontend/src/metabase/query_builder/components/dataref/TablePane.jsx
index e1d0104a659f21941b58590f0e2ae63fdfff238b..16104b9997b4f15f81d946675026aba56a0b56b1 100644
--- a/frontend/src/metabase/query_builder/components/dataref/TablePane.jsx
+++ b/frontend/src/metabase/query_builder/components/dataref/TablePane.jsx
@@ -120,7 +120,7 @@ export default class TablePane extends Component {
                             name="Metrics"
                             type="metrics"
                             show={this.props.show.bind(null, "metric")}
-                            items={table.metrics}
+                            items={table.metrics.filter((metric) => metric.is_active === true)}
                         />
                     }
                     { table.segments && (table.segments.length > 0) &&
@@ -128,7 +128,7 @@ export default class TablePane extends Component {
                             name="Segments"
                             type="segments"
                             show={this.props.show.bind(null, "segment")}
-                            items={table.segments}
+                            items={table.segments.filter((segment) => segment.is_active === true)}
                         />
                     }
                     <div className="Button-group Button-group--brand text-uppercase">
diff --git a/frontend/src/metabase/query_builder/components/filters/FilterList.jsx b/frontend/src/metabase/query_builder/components/filters/FilterList.jsx
index ffeee79882181b31b0d060e118e714466e7254fc..e09fda0b303597d6e9c2b476ae96bcfac0e5ded1 100644
--- a/frontend/src/metabase/query_builder/components/filters/FilterList.jsx
+++ b/frontend/src/metabase/query_builder/components/filters/FilterList.jsx
@@ -7,13 +7,17 @@ import FilterWidget from './FilterWidget.jsx';
 
 import StructuredQuery from "metabase-lib/lib/queries/StructuredQuery";
 import type { Filter } from "metabase/meta/types/Query";
+import Dimension from "metabase-lib/lib/Dimension";
+
+import type { TableMetadata } from "metabase/meta/types/Metadata";
 
 type Props = {
     query: StructuredQuery,
     filters: Array<Filter>,
     removeFilter?: (index: number) => void,
     updateFilter?: (index: number, filter: Filter) => void,
-    maxDisplayValues?: number
+    maxDisplayValues?: number,
+    tableMetadata?: TableMetadata // legacy parameter
 };
 
 type State = {
@@ -49,15 +53,18 @@ export default class FilterList extends Component {
     }
 
     render() {
-        const { query, filters } = this.props;
+        const { query, filters, tableMetadata } = this.props;
         return (
             <div className="Query-filterList scroll-x scroll-show scroll-show--horizontal">
                 {filters.map((filter, index) =>
                     <FilterWidget
                         key={index}
                         placeholder="Item"
-                        // $FlowFixMe: update widgets that are still passing tableMetadata instead of query
-                        query={query || { table: () => this.props.tableMetadata }}
+                        // TODO: update widgets that are still passing tableMetadata instead of query
+                        query={query || {
+                            table: () => tableMetadata,
+                            parseFieldReference: (fieldRef) => Dimension.parseMBQL(fieldRef, tableMetadata)
+                        }}
                         filter={filter}
                         index={index}
                         removeFilter={this.props.removeFilter}
diff --git a/frontend/src/metabase/query_builder/components/template_tags/TagEditorParam.jsx b/frontend/src/metabase/query_builder/components/template_tags/TagEditorParam.jsx
index e29bbd553dbc646226339cc83eebcc1e5f7ffdd1..b622cd89104bdee359e6c6a0b6b2514c4e37a4f9 100644
--- a/frontend/src/metabase/query_builder/components/template_tags/TagEditorParam.jsx
+++ b/frontend/src/metabase/query_builder/components/template_tags/TagEditorParam.jsx
@@ -62,7 +62,7 @@ export default class TagEditorParam extends Component {
             if (!field) {
                 return;
             }
-            const options = parameterOptionsForField(field);
+            const options = parameterOptionsForField(new Field(field));
             let widget_type;
             if (tag.widget_type && _.findWhere(options, { type: tag.widget_type })) {
                 widget_type = tag.widget_type;
@@ -90,7 +90,7 @@ export default class TagEditorParam extends Component {
         if (tag.type === "dimension" && Array.isArray(tag.dimension)) {
             const field = _.findWhere(databaseFields, { id: tag.dimension[1] });
             if (field) {
-                widgetOptions = parameterOptionsForField(field);
+                widgetOptions = parameterOptionsForField(new Field(field));
             }
         }
 
diff --git a/frontend/src/metabase/query_builder/containers/QueryBuilder.integ.spec.js b/frontend/src/metabase/query_builder/containers/QueryBuilder.integ.spec.js
index 6d33dec5c7d4dbaadb13cad646ba3a0cf726831b..50b396b53165b9df73d39277865de9a862ccda46 100644
--- a/frontend/src/metabase/query_builder/containers/QueryBuilder.integ.spec.js
+++ b/frontend/src/metabase/query_builder/containers/QueryBuilder.integ.spec.js
@@ -1,75 +1,162 @@
 import {
     login,
-    globalReduxStore as store,
-    linkContainerToGlobalReduxStore
+    whenOffline,
+    createSavedQuestion,
+    createTestStore,
 } from "metabase/__support__/integrated_tests";
 
 import React from 'react';
-import { parse as urlParse } from "url";
-import { initializeQB } from "../actions";
-import { refreshSiteSettings } from "metabase/redux/settings";
 import QueryBuilder from "metabase/query_builder/containers/QueryBuilder";
 import { mount } from "enzyme";
-import { DATABASE_ID, ORDERS_TABLE_ID, metadata } from "metabase/__support__/sample_dataset_fixture";
-import Question from "metabase-lib/lib/Question";
-import { CardApi } from "metabase/services";
-
-// TODO Atte Keinänen 6/22/17: Write functional mock implementations of Modal and Tooltip
-// We can't use the original classes because they do DOM mutation
-
-jest.mock("metabase/components/Modal", () => {
-    const MockedModal = () => <div className="mocked-modal" />
-    return MockedModal
-});
-
-jest.mock("metabase/components/Tooltip", () => {
-    const MockedTooltip = () => <div className="mocked-tooltip" />
-    return MockedTooltip
-});
-
-const getQBContainer = (cardId) =>
-    linkContainerToGlobalReduxStore(<QueryBuilder params={cardId ? { cardId } : {}} location={{ query: {} }}/>)
-
-const createSavedQuestion = async () => {
-    let unsavedQuestion = Question.create({databaseId: DATABASE_ID, tableId: ORDERS_TABLE_ID, metadata})
-        .query()
-        .addAggregation(["count"])
-        .question()
-
-    unsavedQuestion._card = { ...unsavedQuestion._card, name: "Order count" }
+import {
+    ORDERS_TOTAL_FIELD_ID,
+    unsavedOrderCountQuestion
+} from "metabase/__support__/sample_dataset_fixture";
+import { CANCEL_QUERY, INITIALIZE_QB, QUERY_COMPLETED, QUERY_ERRORED, RUN_QUERY } from "metabase/query_builder/actions";
+import VisualizationError from "metabase/query_builder/components/VisualizationError";
 
-    const savedCard = await CardApi.create(unsavedQuestion.card())
-    return unsavedQuestion.setCard(savedCard);
-}
+import { VisualizationEmptyState } from "metabase/query_builder/components/QueryVisualization";
+import Visualization from "metabase/visualizations/components/Visualization";
+import RunButton from "metabase/query_builder/components/RunButton";
+import { SET_ERROR_PAGE } from "metabase/redux/app";
+import QueryHeader from "metabase/query_builder/components/QueryHeader";
 
 describe("QueryBuilder", () => {
     beforeAll(async () => {
-        await login();
+        await login()
     })
 
     /**
      * Simple tests for seeing if the query builder renders without errors
      */
-
     describe("for new questions", async () => {
-        it("should render normally on page load", async () => {
-            const location = urlParse("/question")
-            await store.dispatch(refreshSiteSettings());
-            await store.dispatch(initializeQB(location, {}))
+        it("renders normally on page load", async () => {
+            const store = await createTestStore()
 
-            // If mount completes without errors, the test will pass
-            mount(getQBContainer());
+            store.pushPath("/question");
+            const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+            await store.waitForActions([INITIALIZE_QB]);
+
+            expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe("New question")
+            expect(qbWrapper.find(VisualizationEmptyState).length).toBe(1)
         });
     });
 
     describe("for saved questions", async () => {
-        it("should render normally on page load", async () => {
-            const question = await createSavedQuestion()
-            const location = urlParse(`/question/${question.id()}`)
-            await store.dispatch(initializeQB(location, {cardId: question.id()}))
+        let savedQuestion = null;
+        beforeAll(async () => {
+            savedQuestion = await createSavedQuestion(unsavedOrderCountQuestion)
+        })
+
+        it("renders normally on page load", async () => {
+            const store = await createTestStore()
+            store.pushPath(savedQuestion.getUrl(savedQuestion));
+            const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+
+            await store.waitForActions([INITIALIZE_QB, QUERY_COMPLETED]);
+            expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe(savedQuestion.displayName())
+        });
+        it("shows an error page if the server is offline", async () => {
+            const store = await createTestStore()
+
+            await whenOffline(async () => {
+                store.pushPath(savedQuestion.getUrl());
+                mount(store.connectContainer(<QueryBuilder />));
+                // only test here that the error page action is dispatched
+                // (it is set on the root level of application React tree)
+                await store.waitForActions([INITIALIZE_QB, SET_ERROR_PAGE]);
+            })
+        })
+        it("doesn't execute the query if user cancels it", async () => {
+            const store = await createTestStore()
+            store.pushPath(savedQuestion.getUrl());
+            const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+            await store.waitForActions([INITIALIZE_QB, RUN_QUERY]);
+
+            const runButton = qbWrapper.find(RunButton);
+            expect(runButton.text()).toBe("Cancel");
+            expect(runButton.simulate("click"));
+
+            await store.waitForActions([CANCEL_QUERY, QUERY_ERRORED]);
+            expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe(savedQuestion.displayName())
+            expect(qbWrapper.find(VisualizationEmptyState).length).toBe(1)
+        })
+    });
+
+
+    describe("for dirty questions", async () => {
+        describe("without original saved question", () => {
+            it("renders normally on page load", async () => {
+                const store = await createTestStore()
+                store.pushPath(unsavedOrderCountQuestion.getUrl());
+                const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+                await store.waitForActions([INITIALIZE_QB, QUERY_COMPLETED]);
+
+                expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe("New question")
+                expect(qbWrapper.find(Visualization).length).toBe(1)
+            });
+            it("fails with a proper error message if the query is invalid", async () => {
+                const invalidQuestion = unsavedOrderCountQuestion.query()
+                    .addBreakout(["datetime-field", ["field-id", 12345], "day"])
+                    .question();
+
+                const store = await createTestStore()
+                store.pushPath(invalidQuestion.getUrl());
+                const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+                await store.waitForActions([INITIALIZE_QB, QUERY_COMPLETED]);
+
+                // TODO: How to get rid of the delay? There is asynchronous initialization in some of VisualizationError parent components
+                // Making the delay shorter causes Jest test runner to crash, see https://stackoverflow.com/a/44075568
+                expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe("New question")
+                expect(qbWrapper.find(VisualizationError).length).toBe(1)
+                expect(qbWrapper.find(VisualizationError).text().includes("There was a problem with your question")).toBe(true)
+            });
+            it("fails with a proper error message if the server is offline", async () => {
+                const store = await createTestStore()
+
+                await whenOffline(async () => {
+                    store.pushPath(unsavedOrderCountQuestion.getUrl());
+                    const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+                    await store.waitForActions([INITIALIZE_QB, QUERY_ERRORED]);
+
+                    expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe("New question")
+                    expect(qbWrapper.find(VisualizationError).length).toBe(1)
+                    expect(qbWrapper.find(VisualizationError).text().includes("We're experiencing server issues")).toBe(true)
+                })
+            })
+            it("doesn't execute the query if user cancels it", async () => {
+                const store = await createTestStore()
+                store.pushPath(unsavedOrderCountQuestion.getUrl());
+                const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+                await store.waitForActions([INITIALIZE_QB, RUN_QUERY]);
+
+                const runButton = qbWrapper.find(RunButton);
+                expect(runButton.text()).toBe("Cancel");
+                expect(runButton.simulate("click"));
+
+                await store.waitForActions([CANCEL_QUERY, QUERY_ERRORED]);
+                expect(qbWrapper.find(QueryHeader).find("h1").text()).toBe("New question")
+                expect(qbWrapper.find(VisualizationEmptyState).length).toBe(1)
+            })
+        })
+        describe("with original saved question", () => {
+            it("should render normally on page load", async () => {
+                const store = await createTestStore()
+                const savedQuestion = await createSavedQuestion(unsavedOrderCountQuestion);
+
+                const dirtyQuestion = savedQuestion
+                    .query()
+                    .addBreakout(["field-id", ORDERS_TOTAL_FIELD_ID])
+                    .question()
+
+                store.pushPath(dirtyQuestion.getUrl(savedQuestion));
+                const qbWrapper = mount(store.connectContainer(<QueryBuilder />));
+                await store.waitForActions([INITIALIZE_QB, QUERY_COMPLETED]);
 
-            // If mount completes without errors, the test will pass
-            mount(getQBContainer(question.id()));
+                const title = qbWrapper.find(QueryHeader).find("h1")
+                expect(title.text()).toBe("New question")
+                expect(title.parent().children().at(1).text()).toBe(`started from ${savedQuestion.displayName()}`)
+            });
         });
     });
 });
diff --git a/frontend/src/metabase/query_builder/reducers.js b/frontend/src/metabase/query_builder/reducers.js
index 854f485a648a035c228f218b03e9cb25dfc408f2..0365d2801a01375ab0a9bf3075fbc4b4c244f893 100644
--- a/frontend/src/metabase/query_builder/reducers.js
+++ b/frontend/src/metabase/query_builder/reducers.js
@@ -154,8 +154,8 @@ export const queryResults = handleActions({
 }, null);
 
 // promise used for tracking a query execution in progress.  when a query is started we capture this.
-export const queryExecutionPromise = handleActions({
-    [RUN_QUERY]: { next: (state, { cancelQueryDeferred }) => cancelQueryDeferred},
+export const cancelQueryDeferred = handleActions({
+    [RUN_QUERY]: { next: (state, { payload: { cancelQueryDeferred } }) => cancelQueryDeferred},
     [CANCEL_QUERY]: { next: (state, { payload }) => null},
     [QUERY_COMPLETED]: { next: (state, { payload }) => null},
     [QUERY_ERRORED]: { next: (state, { payload }) => null},
diff --git a/frontend/src/metabase/questions/collections.js b/frontend/src/metabase/questions/collections.js
index 592417105a325b7ba5ad1890a68ff19bc568216d..faa4142886464e036789768f8888776325b583b6 100644
--- a/frontend/src/metabase/questions/collections.js
+++ b/frontend/src/metabase/questions/collections.js
@@ -2,7 +2,6 @@
 import { createAction, createThunkAction, handleActions, combineReducers } from "metabase/lib/redux";
 import { reset } from 'redux-form';
 import { replace } from "react-router-redux";
-import * as Urls from "metabase/lib/urls";
 
 import _ from "underscore";
 
@@ -36,9 +35,9 @@ export const saveCollection = createThunkAction(SAVE_COLLECTION, (collection) =>
             }
             if (response.id != null) {
                 dispatch(reset("collection"));
+                // use `replace` so form url doesn't appear in history
+                dispatch(replace('/questions/'));
             }
-            // use `replace` so form url doesn't appear in history
-            dispatch(replace(Urls.collection(response)));
             return response;
         } catch (e) {
             // redux-form expects an object with either { field: error } or { _error: error }
diff --git a/frontend/src/metabase/questions/components/ActionHeader.jsx b/frontend/src/metabase/questions/components/ActionHeader.jsx
index a0eb0cea3e13c5b3c2f265158ebbaab2374e0411..ad1c6ea6e22d1860b20fda498867c40a959da470 100644
--- a/frontend/src/metabase/questions/components/ActionHeader.jsx
+++ b/frontend/src/metabase/questions/components/ActionHeader.jsx
@@ -14,13 +14,14 @@ import LabelPopover from "../containers/LabelPopover.jsx";
 const ActionHeader = ({ visibleCount, selectedCount, allAreSelected, sectionIsArchive, setAllSelected, setArchived, labels }) =>
     <div className={S.actionHeader}>
         <Tooltip tooltip={"Select all " + visibleCount} isEnabled={!allAreSelected}>
-            <StackedCheckBox
-                checked={allAreSelected}
-                className="ml1"
-                onChange={(e) => setAllSelected(e.target.checked)}
-                size={20} padding={3} borderColor="currentColor"
-                invertChecked
-            />
+            <span className="ml1">
+                <StackedCheckBox
+                    checked={allAreSelected}
+                    onChange={e => setAllSelected(e.target.checked)}
+                    size={20}
+                    padding={3}
+                />
+            </span>
         </Tooltip>
         <span className={S.selectedCount}>
             {selectedCount} selected
diff --git a/frontend/src/metabase/questions/components/Item.jsx b/frontend/src/metabase/questions/components/Item.jsx
index 93e4450e306d2101a5215889dfa9962e17ac1d4e..28266b96e969c28aba247a1c0a91e4aaf7081fd7 100644
--- a/frontend/src/metabase/questions/components/Item.jsx
+++ b/frontend/src/metabase/questions/components/Item.jsx
@@ -36,19 +36,18 @@ const Item = ({
                     />
                 }
                 { setItemSelected &&
-                    <CheckBox
-                        className={cx(
-                            "cursor-pointer absolute top left",
-                            { "visible text-brand": selected },
-                            { "hover-child text-brand-hover text-light-blue transition-color": !selected }
-                        )}
-                        checked={selected}
-                        onChange={(e) => setItemSelected({ [id]: e.target.checked })}
-                        size={ITEM_ICON_SIZE}
-                        padding={3}
-                        borderColor="currentColor"
-                        invertChecked
-                    />
+                    <span className={cx(
+                        "absolute top left",
+                        { "visible": selected },
+                        { "hover-child": !selected }
+                    )}>
+                        <CheckBox
+                            checked={selected}
+                            onChange={e => setItemSelected({ [id]: e.target.checked })}
+                            size={ITEM_ICON_SIZE}
+                            padding={3}
+                        />
+                    </span>
                 }
             </div>
             <ItemBody
diff --git a/frontend/src/metabase/questions/questions.js b/frontend/src/metabase/questions/questions.js
index 02d4924030df1b86d66269c3a158cad335a3a3b5..b09640bddec913bc42bef782203b8f23f0accb47 100644
--- a/frontend/src/metabase/questions/questions.js
+++ b/frontend/src/metabase/questions/questions.js
@@ -26,7 +26,7 @@ const card = new schema.Entity('cards', {
 
 import { CardApi, CollectionsApi } from "metabase/services";
 
-const LOAD_ENTITIES = 'metabase/questions/LOAD_ENTITIES';
+export const LOAD_ENTITIES = 'metabase/questions/LOAD_ENTITIES';
 const SET_SEARCH_TEXT = 'metabase/questions/SET_SEARCH_TEXT';
 const SET_ITEM_SELECTED = 'metabase/questions/SET_ITEM_SELECTED';
 const SET_ALL_SELECTED = 'metabase/questions/SET_ALL_SELECTED';
diff --git a/frontend/src/metabase/redux/metadata.integ.spec.js b/frontend/src/metabase/redux/metadata.integ.spec.js
index 6ea3d7aa00640440ad617b048beb24326d869dbb..b27bb610441eb00b2b53ddf3a20cbf41073f46ae 100644
--- a/frontend/src/metabase/redux/metadata.integ.spec.js
+++ b/frontend/src/metabase/redux/metadata.integ.spec.js
@@ -6,7 +6,7 @@
  */
 import { getMetadata } from "metabase/selectors/metadata"
 import {
-    createReduxStore,
+    createTestStore,
     login,
 } from "metabase/__support__/integrated_tests";
 import {
@@ -23,9 +23,11 @@ describe("metadata/redux", () => {
     });
 
     describe("METRIC ACTIONS", () => {
+        // TODO Atte Keinänen 6/23/17: Remove metrics after their creation in other tests
         describe("fetchMetrics()", () => {
+            pending();
             it("fetches no metrics in empty db", async () => {
-                const store = createReduxStore();
+                const store = createTestStore();
                 await store.dispatch(fetchMetrics());
                 expect(metadata(store).metricsList().length).toBe(0)
             })
@@ -55,8 +57,11 @@ describe("metadata/redux", () => {
 
     describe("DATABASE ACTIONS", () => {
         describe("fetchDatabases()", () => {
+            // TODO Atte Keinänen 6/23/17: Figure out why on CI two databases show up but locally only one
+            pending();
             it("fetches the sample dataset", async () => {
-                const store = createReduxStore();
+
+                const store = createTestStore();
                 expect(metadata(store).tablesList().length).toBe(0);
                 expect(metadata(store).databasesList().length).toBe(0);
 
@@ -76,9 +81,11 @@ describe("metadata/redux", () => {
 
     describe("TABLE ACTIONS", () => {
         describe("fetchTables()", () => {
+            // TODO Atte Keinänen 6/23/17: Figure out why on CI two databases show up but locally only one
+            pending();
             it("fetches the sample dataset tables", async () => {
                 // what is the difference between fetchDatabases and fetchTables?
-                const store = createReduxStore();
+                const store = createTestStore();
                 expect(metadata(store).tablesList().length).toBe(0);
                 expect(metadata(store).databasesList().length).toBe(0);
 
diff --git a/frontend/src/metabase/redux/metadata.js b/frontend/src/metabase/redux/metadata.js
index 0e7de82de78ac3e077062285fda5af476d02e08a..ff2532bac0e9f81084e4a9068d575ddf2b4ed280 100644
--- a/frontend/src/metabase/redux/metadata.js
+++ b/frontend/src/metabase/redux/metadata.js
@@ -17,7 +17,7 @@ import _ from "underscore";
 
 import { MetabaseApi, MetricApi, SegmentApi, RevisionsApi } from "metabase/services";
 
-const FETCH_METRICS = "metabase/metadata/FETCH_METRICS";
+export const FETCH_METRICS = "metabase/metadata/FETCH_METRICS";
 export const fetchMetrics = createThunkAction(FETCH_METRICS, (reload = false) => {
     return async (dispatch, getState) => {
         const requestStatePath = ["metadata", "metrics"];
@@ -82,7 +82,7 @@ export const updateMetricImportantFields = createThunkAction(UPDATE_METRIC_IMPOR
 });
 
 
-const FETCH_SEGMENTS = "metabase/metadata/FETCH_SEGMENTS";
+export const FETCH_SEGMENTS = "metabase/metadata/FETCH_SEGMENTS";
 export const fetchSegments = createThunkAction(FETCH_SEGMENTS, (reload = false) => {
     return async (dispatch, getState) => {
         const requestStatePath = ["metadata", "segments"];
@@ -125,7 +125,7 @@ export const updateSegment = createThunkAction(UPDATE_SEGMENT, function(segment)
     };
 });
 
-const FETCH_DATABASES = "metabase/metadata/FETCH_DATABASES";
+export const FETCH_DATABASES = "metabase/metadata/FETCH_DATABASES";
 export const fetchDatabases = createThunkAction(FETCH_DATABASES, (reload = false) => {
     return async (dispatch, getState) => {
         const requestStatePath = ["metadata", "databases"];
@@ -146,7 +146,7 @@ export const fetchDatabases = createThunkAction(FETCH_DATABASES, (reload = false
     };
 });
 
-const FETCH_DATABASE_METADATA = "metabase/metadata/FETCH_DATABASE_METADATA";
+export const FETCH_DATABASE_METADATA = "metabase/metadata/FETCH_DATABASE_METADATA";
 export const fetchDatabaseMetadata = createThunkAction(FETCH_DATABASE_METADATA, function(dbId, reload = false) {
     return async function(dispatch, getState) {
         const requestStatePath = ["metadata", "databases", dbId];
@@ -234,7 +234,7 @@ export const fetchTables = createThunkAction(FETCH_TABLES, (reload = false) => {
     };
 });
 
-const FETCH_TABLE_METADATA = "metabase/metadata/FETCH_TABLE_METADATA";
+export const FETCH_TABLE_METADATA = "metabase/metadata/FETCH_TABLE_METADATA";
 export const fetchTableMetadata = createThunkAction(FETCH_TABLE_METADATA, function(tableId, reload = false) {
     return async function(dispatch, getState) {
         const requestStatePath = ["metadata", "tables", tableId];
@@ -302,7 +302,7 @@ export const updateField = createThunkAction(UPDATE_FIELD, function(field) {
     };
 });
 
-const FETCH_REVISIONS = "metabase/metadata/FETCH_REVISIONS";
+export const FETCH_REVISIONS = "metabase/metadata/FETCH_REVISIONS";
 export const fetchRevisions = createThunkAction(FETCH_REVISIONS, (type, id, reload = false) => {
     return async (dispatch, getState) => {
         const requestStatePath = ["metadata", "revisions", type, id];
@@ -327,7 +327,7 @@ export const fetchRevisions = createThunkAction(FETCH_REVISIONS, (type, id, relo
 });
 
 // for fetches with data dependencies in /reference
-const FETCH_METRIC_TABLE = "metabase/metadata/FETCH_METRIC_TABLE";
+export const FETCH_METRIC_TABLE = "metabase/metadata/FETCH_METRIC_TABLE";
 export const fetchMetricTable = createThunkAction(FETCH_METRIC_TABLE, (metricId, reload = false) => {
     return async (dispatch, getState) => {
         await dispatch(fetchMetrics()); // FIXME: fetchMetric?
@@ -337,7 +337,7 @@ export const fetchMetricTable = createThunkAction(FETCH_METRIC_TABLE, (metricId,
     };
 });
 
-const FETCH_METRIC_REVISIONS = "metabase/metadata/FETCH_METRIC_REVISIONS";
+export const FETCH_METRIC_REVISIONS = "metabase/metadata/FETCH_METRIC_REVISIONS";
 export const fetchMetricRevisions = createThunkAction(FETCH_METRIC_REVISIONS, (metricId, reload = false) => {
     return async (dispatch, getState) => {
         await Promise.all([
@@ -350,7 +350,7 @@ export const fetchMetricRevisions = createThunkAction(FETCH_METRIC_REVISIONS, (m
     };
 });
 
-const FETCH_SEGMENT_FIELDS = "metabase/metadata/FETCH_SEGMENT_FIELDS";
+export const FETCH_SEGMENT_FIELDS = "metabase/metadata/FETCH_SEGMENT_FIELDS";
 export const fetchSegmentFields = createThunkAction(FETCH_SEGMENT_FIELDS, (segmentId, reload = false) => {
     return async (dispatch, getState) => {
         await dispatch(fetchSegments()); // FIXME: fetchSegment?
@@ -363,7 +363,7 @@ export const fetchSegmentFields = createThunkAction(FETCH_SEGMENT_FIELDS, (segme
     };
 });
 
-const FETCH_SEGMENT_TABLE = "metabase/metadata/FETCH_SEGMENT_TABLE";
+export const FETCH_SEGMENT_TABLE = "metabase/metadata/FETCH_SEGMENT_TABLE";
 export const fetchSegmentTable = createThunkAction(FETCH_SEGMENT_TABLE, (segmentId, reload = false) => {
     return async (dispatch, getState) => {
         await dispatch(fetchSegments()); // FIXME: fetchSegment?
@@ -373,7 +373,7 @@ export const fetchSegmentTable = createThunkAction(FETCH_SEGMENT_TABLE, (segment
     };
 });
 
-const FETCH_SEGMENT_REVISIONS = "metabase/metadata/FETCH_SEGMENT_REVISIONS";
+export const FETCH_SEGMENT_REVISIONS = "metabase/metadata/FETCH_SEGMENT_REVISIONS";
 export const fetchSegmentRevisions = createThunkAction(FETCH_SEGMENT_REVISIONS, (segmentId, reload = false) => {
     return async (dispatch, getState) => {
         await Promise.all([
diff --git a/frontend/src/metabase/reference/components/EditableReferenceHeader.jsx b/frontend/src/metabase/reference/components/EditableReferenceHeader.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..4e7cbb0e52d9e226ed70ead35deed2d3ee1443e0
--- /dev/null
+++ b/frontend/src/metabase/reference/components/EditableReferenceHeader.jsx
@@ -0,0 +1,134 @@
+import React from "react";
+import PropTypes from "prop-types";
+import { Link } from "react-router";
+import cx from "classnames";
+import pure from "recompose/pure";
+
+import S from "./ReferenceHeader.css";
+import L from "metabase/components/List.css";
+import E from "metabase/reference/components/EditButton.css";
+
+import IconBorder from "metabase/components/IconBorder.jsx";
+import Icon from "metabase/components/Icon.jsx";
+import Ellipsified from "metabase/components/Ellipsified.jsx";
+import EditButton from "metabase/reference/components/EditButton.jsx";
+
+
+const EditableReferenceHeader = ({
+    entity = {},
+    table,
+    type,
+    headerIcon,
+    headerLink,
+    name,
+    user,
+    isEditing,
+    hasSingleSchema,
+    hasDisplayName,
+    startEditing,
+    displayNameFormField,
+    nameFormField
+}) =>
+    <div className="wrapper wrapper--trim">
+        <div className={cx("relative", L.header)} style={type === 'segment' ? {marginBottom: 0} : {}}>
+            <div className={L.leftIcons}>
+                { headerIcon &&
+                    <IconBorder
+                        borderWidth="0"
+                        style={{backgroundColor: "#E9F4F8"}}
+                    >
+                        <Icon
+                            className="text-brand"
+                            name={headerIcon}
+                            width={24}
+                            height={24}
+                        />
+                    </IconBorder>
+                }
+            </div>
+            { type === 'table' && !hasSingleSchema && !isEditing &&
+                <div className={S.headerSchema}>{entity.schema}</div>
+            }
+            <div
+                className={S.headerBody}
+                style={isEditing && name === 'Details' ? {alignItems: "flex-start"} : {}}
+            >
+                { isEditing && name === 'Details' ?
+                    hasDisplayName ?
+                        <input
+                            className={S.headerTextInput}
+                            type="text"
+                            placeholder={entity.name}
+                            {...displayNameFormField}
+                            defaultValue={entity.display_name}
+                        /> :
+                        <input
+                            className={S.headerTextInput}
+                            type="text"
+                            placeholder={entity.name}
+                            {...nameFormField}
+                            defaultValue={entity.name}
+                        /> :
+                    [
+                        <Ellipsified
+                            key="1"
+                            className={!headerLink && "flex-full"}
+                            tooltipMaxWidth="100%"
+                        >
+                            { name === 'Details' ?
+                                hasDisplayName ?
+                                    entity.display_name || entity.name :
+                                    entity.name :
+                                name
+                            }
+                        </Ellipsified>,
+                        headerLink &&
+                            <div key="2" className={cx("flex-full", S.headerButton)}>
+                                <Link
+                                    to={headerLink}
+                                    className={cx("Button", "Button--borderless", "ml3", E.editButton)}
+                                    data-metabase-event={`Data Reference;Entity -> QB click;${type}`}
+                                >
+                                    <div className="flex align-center relative">
+                                        <span className="mr1 flex-no-shrink">See this {type}</span>
+                                        <Icon name="chevronright" size={16} />
+                                    </div>
+                                </Link>
+                            </div>
+                    ]
+                }
+                { user && user.is_superuser && !isEditing &&
+                    <EditButton className="ml1" startEditing={startEditing} />
+                }
+            </div>
+        </div>
+        { type === 'segment' && table &&
+            <div className={S.subheader}>
+                <div className={cx(S.subheaderBody)}>
+                    A subset of <Link
+                        className={S.subheaderLink}
+                        to={`/reference/databases/${table.db_id}/tables/${table.id}`}
+                    >
+                        {table.display_name}
+                    </Link>
+                </div>
+            </div>
+        }
+    </div>;
+EditableReferenceHeader.propTypes = {
+    entity: PropTypes.object,
+    table: PropTypes.object,
+    type: PropTypes.string,
+    headerIcon: PropTypes.string,
+    headerLink: PropTypes.string,
+    name: PropTypes.string,
+    user: PropTypes.object,
+    isEditing: PropTypes.bool,
+    hasSingleSchema: PropTypes.bool,
+    hasDisplayName: PropTypes.bool,
+    startEditing: PropTypes.func,
+    displayNameFormField: PropTypes.object,
+    nameFormField: PropTypes.object
+};
+
+export default pure(EditableReferenceHeader);
diff --git a/frontend/src/metabase/reference/components/FieldsToGroupBy.jsx b/frontend/src/metabase/reference/components/FieldsToGroupBy.jsx
index 48cd1535bfbd2e533b04ade920f101820355a8f3..7781be696a1135925e3337dce4973c19835d4ace 100644
--- a/frontend/src/metabase/reference/components/FieldsToGroupBy.jsx
+++ b/frontend/src/metabase/reference/components/FieldsToGroupBy.jsx
@@ -1,7 +1,6 @@
-import React from "react";
-import PropTypes from "prop-types";
+import React, { Component } from "react";
 import cx from "classnames";
-import pure from "recompose/pure";
+import { connect } from "react-redux";
 
 import S from "./UsefulQuestions.css";
 import D from "metabase/reference/components/Detail.css";
@@ -13,49 +12,64 @@ import {
 
 import FieldToGroupBy from "metabase/reference/components/FieldToGroupBy.jsx";
 
-const FieldsToGroupBy = ({
-    fields,
-    databaseId,
-    metric,
-    title,
-    onChangeLocation
-}) =>
-    <div className={cx(D.detail)}>
-        <div className={D.detailBody}>
-            <div className={D.detailTitle}>
-                <span className={D.detailName}>{title}</span>
-            </div>
-            <div className={S.usefulQuestions}>
-                { fields && Object.values(fields)
-                        .map((field, index, fields) =>
-                            <FieldToGroupBy
-                                key={field.id}
-                                className={cx("border-bottom", "pt1", "pb1")}
-                                iconClass={L.icon}
-                                field={field}
-                                metric={metric}
-                                onClick={() => onChangeLocation(getQuestionUrl({
+import { fetchTableMetadata } from "metabase/redux/metadata";
+import { getMetadata } from "metabase/selectors/metadata";
+import Metadata from "metabase-lib/lib/metadata/Metadata";
+
+const mapDispatchToProps = {
+    fetchTableMetadata,
+};
+
+const mapStateToProps = (state, props) => ({
+    metadata: getMetadata(state, props)
+})
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class FieldsToGroupBy extends Component {
+    props: {
+        fields: Object,
+        databaseId: number,
+        metric: Object,
+        title: string,
+        onChangeLocation: (string) => void,
+        metadata: Metadata
+    }
+
+    render() {
+        const { fields, databaseId, metric, title, onChangeLocation, metadata } = this.props;
+
+        return (
+            <div className={cx(D.detail)}>
+                <div className={D.detailBody}>
+                    <div className={D.detailTitle}>
+                        <span className={D.detailName}>{title}</span>
+                    </div>
+                    <div className={S.usefulQuestions}>
+                        { fields && Object.values(fields)
+                            .map((field, index, fields) =>
+                                <FieldToGroupBy
+                                    key={field.id}
+                                    className={cx("border-bottom", "pt1", "pb1")}
+                                    iconClass={L.icon}
+                                    field={field}
+                                    metric={metric}
+                                    onClick={() => onChangeLocation(getQuestionUrl({
                                         dbId: databaseId,
                                         tableId: field.table_id,
                                         fieldId: field.id,
-                                        metricId: metric.id
+                                        metricId: metric.id,
+                                        metadata
                                     }))}
-                                secondaryOnClick={(event) => {
-                                    event.stopPropagation();
-                                    onChangeLocation(`/reference/databases/${databaseId}/tables/${field.table_id}/fields/${field.id}`);
-                                }}
-                            />
-                        )
-                }
+                                    secondaryOnClick={(event) => {
+                                        event.stopPropagation();
+                                        onChangeLocation(`/reference/databases/${databaseId}/tables/${field.table_id}/fields/${field.id}`);
+                                    }}
+                                />
+                            )
+                        }
+                    </div>
+                </div>
             </div>
-        </div>
-    </div>;
-FieldsToGroupBy.propTypes = {
-    fields: PropTypes.object.isRequired,
-    databaseId: PropTypes.number.isRequired,
-    metric: PropTypes.object.isRequired,
-    title: PropTypes.string.isRequired,
-    onChangeLocation: PropTypes.func.isRequired
-};
-
-export default pure(FieldsToGroupBy);
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/components/Formula.jsx b/frontend/src/metabase/reference/components/Formula.jsx
index 4882a4d1b741312af2c100be4e5c21803db372f2..fad946d8f147dc5f10be18c2c3c487a7aba25a4a 100644
--- a/frontend/src/metabase/reference/components/Formula.jsx
+++ b/frontend/src/metabase/reference/components/Formula.jsx
@@ -1,7 +1,6 @@
-import React from "react";
-import PropTypes from "prop-types";
-import pure from "recompose/pure";
+import React, { Component } from "react";
 import cx from "classnames";
+import { connect } from "react-redux";
 
 import ReactCSSTransitionGroup from "react-addons-css-transition-group";
 
@@ -10,46 +9,58 @@ import S from "./Formula.css";
 import Icon from "metabase/components/Icon.jsx";
 
 import QueryDefinition from "metabase/query_builder/components/dataref/QueryDefinition.jsx";
+import { fetchTableMetadata } from "metabase/redux/metadata";
+import { getMetadata } from "metabase/selectors/metadata";
 
-const Formula = ({
-    type,
-    entity,
-    table,
-    isExpanded,
-    expandFormula,
-    collapseFormula
-}) =>
-    <div
-        className={cx(S.formula)}
-        onClick={isExpanded ? collapseFormula : expandFormula}
-    >
-        <div className={S.formulaHeader}>
-            <Icon name="beaker" size={24} />
-            <span className={S.formulaTitle}>View the {type} formula</span>
-        </div>
-        <ReactCSSTransitionGroup
-            transitionName="formulaDefinition"
-            transitionEnterTimeout={300}
-            transitionLeaveTimeout={300}
-        >
-            { isExpanded &&
-                <div key="formulaDefinition" className="formulaDefinition">
-                    <QueryDefinition
-                        className={S.formulaDefinitionInner}
-                        object={entity}
-                        tableMetadata={table}
-                    />
-                </div>
-            }
-        </ReactCSSTransitionGroup>
-    </div>
-
-Formula.propTypes = {
-    type: PropTypes.string.isRequired,
-    entity: PropTypes.object.isRequired,
-    table: PropTypes.object.isRequired,
-    isExpanded: PropTypes.bool.isRequired,
-    expandFormula: PropTypes.func.isRequired
+import type Metadata from "metabase-lib/lib/metadata/Metadata";
+
+const mapDispatchToProps = {
+    fetchTableMetadata,
 };
 
-export default pure(Formula);
+const mapStateToProps = (state, props) => ({
+    metadata: getMetadata(state, props)
+})
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class Formula extends Component {
+    props: {
+        type: string,
+        entity: Object,
+        isExpanded: boolean,
+        expandFormula: any,
+        collapseFormula: any,
+        metadata: Metadata
+    }
+
+    render() {
+        const { type, entity, isExpanded, expandFormula, collapseFormula, metadata } = this.props;
+
+        return (
+            <div
+                className={cx(S.formula)}
+                onClick={isExpanded ? collapseFormula : expandFormula}
+            >
+                <div className={S.formulaHeader}>
+                    <Icon name="beaker" size={24}/>
+                    <span className={S.formulaTitle}>View the {type} formula</span>
+                </div>
+                <ReactCSSTransitionGroup
+                    transitionName="formulaDefinition"
+                    transitionEnterTimeout={300}
+                    transitionLeaveTimeout={300}
+                >
+                    { isExpanded &&
+                    <div key="formulaDefinition" className="formulaDefinition">
+                        <QueryDefinition
+                            className={S.formulaDefinitionInner}
+                            object={entity}
+                            tableMetadata={metadata.tables[entity.table_id]}
+                        />
+                    </div>
+                    }
+                </ReactCSSTransitionGroup>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/components/ReferenceHeader.jsx b/frontend/src/metabase/reference/components/ReferenceHeader.jsx
index 9a5fcfedde558e2b70d27cc1fa772aa510297e15..fe467edba01f61786c34486a014e17088bdf68d4 100644
--- a/frontend/src/metabase/reference/components/ReferenceHeader.jsx
+++ b/frontend/src/metabase/reference/components/ReferenceHeader.jsx
@@ -11,118 +11,68 @@ import E from "metabase/reference/components/EditButton.css";
 import IconBorder from "metabase/components/IconBorder.jsx";
 import Icon from "metabase/components/Icon.jsx";
 import Ellipsified from "metabase/components/Ellipsified.jsx";
-import EditButton from "metabase/reference/components/EditButton.jsx";
 
 
 const ReferenceHeader = ({
-    entity = {},
-    table,
-    section,
-    user,
-    isEditing,
-    hasSingleSchema,
-    hasDisplayName,
-    startEditing,
-    displayNameFormField,
-    nameFormField
+    name,
+    type,
+    headerIcon,
+    headerBody,
+    headerLink
 }) =>
     <div className="wrapper wrapper--trim">
-        <div className={cx("relative", L.header)} style={section.type === 'segment' ? {marginBottom: 0} : {}}>
+        <div className={cx("relative", L.header)}>
             <div className={L.leftIcons}>
-                { section.headerIcon &&
+                { headerIcon &&
                     <IconBorder
                         borderWidth="0"
                         style={{backgroundColor: "#E9F4F8"}}
                     >
                         <Icon
                             className="text-brand"
-                            name={section.headerIcon}
+                            name={headerIcon}
                             width={24}
                             height={24}
                         />
                     </IconBorder>
                 }
             </div>
-            { section.type === 'table' && !hasSingleSchema && !isEditing &&
-                <div className={S.headerSchema}>{entity.schema}</div>
-            }
             <div
                 className={S.headerBody}
-                style={isEditing && section.name === 'Details' ? {alignItems: "flex-start"} : {}}
             >
-                { isEditing && section.name === 'Details' ?
-                    hasDisplayName ?
-                        <input
-                            className={S.headerTextInput}
-                            type="text"
-                            placeholder={entity.name}
-                            {...displayNameFormField}
-                            defaultValue={entity.display_name}
-                        /> :
-                        <input
-                            className={S.headerTextInput}
-                            type="text"
-                            placeholder={entity.name}
-                            {...nameFormField}
-                            defaultValue={entity.name}
-                        /> :
-                    [
-                        <Ellipsified
-                            key="1"
-                            className={!section.headerLink && "flex-full"}
-                            tooltipMaxWidth="100%"
+                <Ellipsified
+                    key="1"
+                    className={!headerLink && "flex-full"}
+                    tooltipMaxWidth="100%"
+                >
+                    { name }
+                </Ellipsified>
+                
+                {headerLink &&
+                    <div key="2" className={cx("flex-full", S.headerButton)}>
+                        <Link
+                            to={headerLink}
+                            className={cx("Button", "Button--borderless", "ml3", E.editButton)}
+                            data-metabase-event={`Data Reference;Entity -> QB click;${type}`}
                         >
-                            { section.name === 'Details' ?
-                                hasDisplayName ?
-                                    entity.display_name || entity.name :
-                                    entity.name :
-                                section.name
-                            }
-                        </Ellipsified>,
-                        section.headerLink &&
-                            <div key="2" className={cx("flex-full", S.headerButton)}>
-                                <Link
-                                    to={section.headerLink}
-                                    className={cx("Button", "Button--borderless", "ml3", E.editButton)}
-                                    data-metabase-event={`Data Reference;Entity -> QB click;${section.type}`}
-                                >
-                                    <div className="flex align-center relative">
-                                        <span className="mr1 flex-no-shrink">See this {section.type}</span>
-                                        <Icon name="chevronright" size={16} />
-                                    </div>
-                                </Link>
+                            <div className="flex align-center relative">
+                                <span className="mr1 flex-no-shrink">See this {type}</span>
+                                <Icon name="chevronright" size={16} />
                             </div>
-                    ]
-                }
-                { user && user.is_superuser && !isEditing &&
-                    <EditButton className="ml1" startEditing={startEditing} />
+                        </Link>
+                    </div>
                 }
             </div>
         </div>
-        { section.type === 'segment' && table &&
-            <div className={S.subheader}>
-                <div className={cx(S.subheaderBody)}>
-                    A subset of <Link
-                        className={S.subheaderLink}
-                        to={`/reference/databases/${table.db_id}/tables/${table.id}`}
-                    >
-                        {table.display_name}
-                    </Link>
-                </div>
-            </div>
-        }
     </div>;
+
 ReferenceHeader.propTypes = {
-    entity: PropTypes.object,
-    table: PropTypes.object,
-    section: PropTypes.object.isRequired,
-    user: PropTypes.object,
-    isEditing: PropTypes.bool,
-    hasSingleSchema: PropTypes.bool,
-    hasDisplayName: PropTypes.bool,
-    startEditing: PropTypes.func,
-    displayNameFormField: PropTypes.object,
-    nameFormField: PropTypes.object
+    name: PropTypes.string.isRequired,
+    type: PropTypes.string,
+    headerIcon: PropTypes.string,
+    headerBody: PropTypes.string,
+    headerLink: PropTypes.string
+
 };
 
 export default pure(ReferenceHeader);
diff --git a/frontend/src/metabase/reference/databases/DatabaseDetail.jsx b/frontend/src/metabase/reference/databases/DatabaseDetail.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..00d42fcc9d4ef7148dabe0fc6e34b7edf7d3c99d
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/DatabaseDetail.jsx
@@ -0,0 +1,176 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+import { push } from "react-router-redux";
+
+import List from "metabase/components/List.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+import Detail from "metabase/reference/components/Detail.jsx";
+
+import {
+    getDatabase,
+    getTable,
+    getFields,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getIsFormulaExpanded,
+    getForeignKeys
+} from "../selectors";
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+
+const mapStateToProps = (state, props) => {
+    const entity = getDatabase(state, props) || {};
+    const fields = getFields(state, props);
+
+    return {
+        entity,
+        table: getTable(state, props),
+        metadataFields: fields,
+        loading: getLoading(state, props),
+        // naming this 'error' will conflict with redux form
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        foreignKeys: getForeignKeys(state, props),
+        isEditing: getIsEditing(state, props),
+        isFormulaExpanded: getIsFormulaExpanded(state, props),
+    }
+};
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions,
+    onChangeLocation: push
+};
+
+const validate = (values, props) => {
+    return {};
+}
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'details',
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats'],
+    validate
+})
+export default class DatabaseDetail extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entity: PropTypes.object.isRequired,
+        table: PropTypes.object,
+        user: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        resetForm: PropTypes.func.isRequired,
+        fields: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool
+    };
+
+    render() {
+        const {
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats },
+            style,
+            entity,
+            table,
+            loadingError,
+            loading,
+            user,
+            isEditing,
+            startEditing,
+            endEditing,
+            handleSubmit,
+            resetForm,
+            submitting
+        } = this.props;
+
+        const onSubmit = handleSubmit(async (fields) =>
+            await actions.rUpdateDatabaseDetail(fields, this.props)
+        );
+
+        return (
+            <form style={style} className="full"
+                onSubmit={onSubmit}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={false}
+                        onSubmit={onSubmit}
+                        endEditing={endEditing}
+                        reinitializeForm={resetForm}
+                        submitting={submitting}
+                        revisionMessageFormField={revision_message}
+                    />
+                }
+                <EditableReferenceHeader
+                    entity={entity}
+                    table={table}
+                    type="database"
+                    headerIcon="database"
+                    name="Details"
+                    user={user}
+                    isEditing={isEditing}
+                    hasSingleSchema={false}
+                    hasDisplayName={false}
+                    startEditing={startEditing}
+                    displayNameFormField={display_name}
+                    nameFormField={name}
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () =>
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            <li className="relative">
+                                <Detail
+                                    id="description"
+                                    name="Description"
+                                    description={entity.description}
+                                    placeholder="No description yet"
+                                    isEditing={isEditing}
+                                    field={description}
+                                />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="points_of_interest"
+                                    name={`Why this database is interesting`}
+                                    description={entity.points_of_interest}
+                                    placeholder="Nothing interesting yet"
+                                    isEditing={isEditing}
+                                    field={points_of_interest}
+                                    />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="caveats"
+                                    name={`Things to be aware of about this database`}
+                                    description={entity.caveats}
+                                    placeholder="Nothing to be aware of yet"
+                                    isEditing={isEditing}
+                                    field={caveats}
+                                />
+                            </li>
+                        </List>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/DatabaseDetailContainer.jsx b/frontend/src/metabase/reference/databases/DatabaseDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..340f8cd57b9bfb97c3fd4ac2f9cf255597d9bce8
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/DatabaseDetailContainer.jsx
@@ -0,0 +1,74 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import DatabaseSidebar from './DatabaseSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import DatabaseDetail from "metabase/reference/databases/DatabaseDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabase,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    database: getDatabase(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class DatabaseDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabaseMetadata(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            database,
+            isEditing
+        } = this.props;
+
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<DatabaseSidebar database={database} />}
+            >
+                <DatabaseDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/DatabaseList.jsx b/frontend/src/metabase/reference/databases/DatabaseList.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..1aa94382027e87d23bcddc92e4da7e7fae13a868
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/DatabaseList.jsx
@@ -0,0 +1,98 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+
+import { isQueryable } from "metabase/lib/table";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getDatabases,
+    getError,
+    getLoading
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+const emptyStateData = {
+    title: "Metabase is no fun without any data",
+    adminMessage: "Your databases will appear here once you connect one",
+    message: "Databases will appear here once your admins have added some",
+    image: "app/assets/img/databases-list",
+    adminAction: "Connect a database",
+    adminLink: "/admin/databases/create"
+}
+
+const mapStateToProps = (state, props) => ({
+    entities: getDatabases(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class DatabaseList extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name="Databases and tables"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            { 
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                          <li className="relative" key={entity.id}>
+                                            <ListItem
+                                                id={entity.id}
+                                                index={index}
+                                                name={entity.display_name || entity.name}
+                                                description={ entity.description }
+                                                url={ `/reference/databases/${entity.id}` }
+                                                icon="database"
+                                            />
+                                        </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/DatabaseListContainer.jsx b/frontend/src/metabase/reference/databases/DatabaseListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..c74fbefbc9970444ed919adaff5a82aa54ae0fa9
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/DatabaseListContainer.jsx
@@ -0,0 +1,69 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import BaseSidebar from 'metabase/reference/guide/BaseSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import DatabaseList from "metabase/reference/databases/DatabaseList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+const mapStateToProps = (state, props) => ({
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class DatabaseListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        location: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabases(this.props);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+
+    }
+
+    render() {
+        const {
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<BaseSidebar/>}
+            >
+                <DatabaseList {...this.props}/>
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/DatabaseSidebar.jsx b/frontend/src/metabase/reference/databases/DatabaseSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..9ef1dd1adba36db281bb6f96000c2fb22f399d65
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/DatabaseSidebar.jsx
@@ -0,0 +1,45 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const DatabaseSidebar = ({
+    database,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <ul>
+            <div className={S.breadcrumbs}>
+                <Breadcrumbs
+                    className="py4"
+                    crumbs={[["Databases","/reference/databases"],
+                             [database.name]]}
+                    inSidebar={true}
+                    placeholder="Data Reference"
+                />
+            </div>
+                <SidebarItem key={`/reference/databases/${database.id}`} 
+                             href={`/reference/databases/${database.id}`} 
+                             icon="document" 
+                             name="Details" />
+                <SidebarItem key={`/reference/databases/${database.id}/tables`} 
+                             href={`/reference/databases/${database.id}/tables`} 
+                             icon="table2" 
+                             name={`Tables in ${database.name}`} />
+        </ul>
+    </div>
+DatabaseSidebar.propTypes = {
+    database:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(DatabaseSidebar);
+
diff --git a/frontend/src/metabase/reference/databases/FieldDetail.jsx b/frontend/src/metabase/reference/databases/FieldDetail.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..940193202323b717a3b2d52f9fa541953eb92968
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/FieldDetail.jsx
@@ -0,0 +1,261 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+import { push } from "react-router-redux";
+
+import S from "metabase/reference/Reference.css";
+
+import List from "metabase/components/List.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+import Detail from "metabase/reference/components/Detail.jsx";
+import FieldTypeDetail from "metabase/reference/components/FieldTypeDetail.jsx";
+import UsefulQuestions from "metabase/reference/components/UsefulQuestions.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+import {
+    getField,
+    getTable,
+    getDatabase,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getIsFormulaExpanded,
+    getForeignKeys
+} from "../selectors";
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+
+const interestingQuestions = (database, table, field) => {
+    return [
+        {
+            text: `Number of ${table.display_name} grouped by ${field.display_name}`,
+            icon: { name: "bar", scale: 1, viewBox: "8 8 16 16" },
+            link: getQuestionUrl({
+                dbId: database.id,
+                tableId: table.id,
+                fieldId: field.id,
+                getCount: true,
+                visualization: 'bar'
+            })
+        },
+        {
+            text: `Number of ${table.display_name} grouped by ${field.display_name}`,
+            icon: { name: "pie", scale: 1, viewBox: "8 8 16 16" },
+            link: getQuestionUrl({
+                dbId: database.id,
+                tableId: table.id,
+                fieldId: field.id,
+                getCount: true,
+                visualization: 'pie'
+            })
+        },
+        {
+            text: `All distinct values of ${field.display_name}`,
+            icon: "table2",
+            link: getQuestionUrl({
+                dbId: database.id,
+                tableId: table.id,
+                fieldId: field.id
+            })
+        }
+    ]
+}
+
+const mapStateToProps = (state, props) => {
+    const entity = getField(state, props) || {};
+
+    return {
+        entity,
+        field: entity,
+        table: getTable(state, props),
+        database: getDatabase(state, props),
+        loading: getLoading(state, props),
+        // naming this 'error' will conflict with redux form
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        foreignKeys: getForeignKeys(state, props),
+        isEditing: getIsEditing(state, props),
+        isFormulaExpanded: getIsFormulaExpanded(state, props),
+    }
+};
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions,
+    onChangeLocation: push
+};
+
+const validate = (values, props) => {
+    return {};
+}
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'details',
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats', 'special_type', 'fk_target_field_id'],
+    validate
+})
+export default class FieldDetail extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entity: PropTypes.object.isRequired,
+        field:  PropTypes.object.isRequired,
+        table: PropTypes.object,
+        user: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        foreignKeys: PropTypes.object,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        resetForm: PropTypes.func.isRequired,
+        fields: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool,
+    };
+
+    render() {
+        const {
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats, special_type, fk_target_field_id },
+            style,
+            entity,
+            table,
+            loadingError,
+            loading,
+            user,
+            foreignKeys,
+            isEditing,
+            startEditing,
+            endEditing,
+            handleSubmit,
+            resetForm,
+            submitting,
+        } = this.props;
+
+        const onSubmit = handleSubmit(async (fields) =>
+            await actions.rUpdateFieldDetail(fields, this.props)
+        );
+
+        return (
+            <form style={style} className="full"
+                onSubmit={onSubmit}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={false}
+                        onSubmit={onSubmit}
+                        endEditing={endEditing}
+                        reinitializeForm={resetForm}
+                        submitting={submitting}
+                        revisionMessageFormField={revision_message}
+                    />
+                }
+                <EditableReferenceHeader
+                    entity={entity}
+                    table={table}
+                    type="field"
+                    headerIcon="field"
+                    name="Details"
+                    user={user}
+                    isEditing={isEditing}
+                    hasSingleSchema={false}
+                    hasDisplayName={true}
+                    startEditing={startEditing}
+                    displayNameFormField={display_name}
+                    nameFormField={name}
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () =>
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            <li className="relative">
+                                <Detail
+                                    id="description"
+                                    name="Description"
+                                    description={entity.description}
+                                    placeholder="No description yet"
+                                    isEditing={isEditing}
+                                    field={description}
+                                />
+                            </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <Detail
+                                        id="name"
+                                        name="Actual name in database"
+                                        description={entity.name}
+                                        subtitleClass={S.tableActualName}
+                                    />
+                                </li>
+                            }
+                            <li className="relative">
+                                <Detail
+                                    id="points_of_interest"
+                                    name={`Why this field is interesting`}
+                                    description={entity.points_of_interest}
+                                    placeholder="Nothing interesting yet"
+                                    isEditing={isEditing}
+                                    field={points_of_interest}
+                                    />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="caveats"
+                                    name={`Things to be aware of about this field`}
+                                    description={entity.caveats}
+                                    placeholder="Nothing to be aware of yet"
+                                    isEditing={isEditing}
+                                    field={caveats}
+                                />
+                            </li>
+
+
+                            { !isEditing && 
+                                <li className="relative">
+                                    <Detail
+                                        id="base_type"
+                                        name={`Data type`}
+                                        description={entity.base_type}
+                                    />
+                                </li>
+                            }
+                                <li className="relative">
+                                    <FieldTypeDetail
+                                        field={entity}
+                                        foreignKeys={foreignKeys}
+                                        fieldTypeFormField={special_type}
+                                        foreignKeyFormField={fk_target_field_id}
+                                        isEditing={isEditing}
+                                    />
+                                </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <UsefulQuestions questions={interestingQuestions(this.props.database, this.props.table, this.props.field)} />
+                                </li>
+                            }
+
+
+                        </List>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/FieldDetailContainer.jsx b/frontend/src/metabase/reference/databases/FieldDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..37d5fabcb9802dbb305f04db060b2c0bbf3ab576
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/FieldDetailContainer.jsx
@@ -0,0 +1,81 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import FieldSidebar from './FieldSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import FieldDetail from "metabase/reference/databases/FieldDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabase,
+    getTable,
+    getField,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+const mapStateToProps = (state, props) => ({
+    database: getDatabase(state, props),    
+    table: getTable(state, props),    
+    field: getField(state, props),    
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class FieldDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        table: PropTypes.object.isRequired,
+        field: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabaseMetadata(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            database,
+            table,
+            field,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<FieldSidebar database={database} table={table} field={field}/>}
+            >
+                <FieldDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/containers/ReferenceFieldsList.jsx b/frontend/src/metabase/reference/databases/FieldList.jsx
similarity index 73%
rename from frontend/src/metabase/reference/containers/ReferenceFieldsList.jsx
rename to frontend/src/metabase/reference/databases/FieldList.jsx
index a5aebda8df7c32abe0e925d300af48dd7f7493f4..5f68fb113e336b4a264b64bc0c280ccbaff77dd3 100644
--- a/frontend/src/metabase/reference/containers/ReferenceFieldsList.jsx
+++ b/frontend/src/metabase/reference/databases/FieldList.jsx
@@ -14,23 +14,21 @@ import EmptyState from "metabase/components/EmptyState.jsx";
 import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
 
 import EditHeader from "metabase/reference/components/EditHeader.jsx";
-import ReferenceHeader from "metabase/reference/components/ReferenceHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
 
 import cx from "classnames";
 
 import {
-    getSection,
-    getData,
+    getTable,
+    getFieldsByTable,
     getForeignKeys,
     getError,
     getLoading,
     getUser,
     getIsEditing,
-    getHasRevisionHistory,
 } from "../selectors";
 
 import {
-    tryUpdateFields,
     fieldsToFormFields
 } from '../utils';
 
@@ -39,17 +37,23 @@ import { getIconForField } from "metabase/lib/schema_metadata";
 import * as metadataActions from "metabase/redux/metadata";
 import * as actions from 'metabase/reference/reference';
 
+
+const emptyStateData = {
+    message: `Fields in this table will appear here as they're added`,
+    icon: "fields"
+}
+
+
 const mapStateToProps = (state, props) => {
-    const data = getData(state, props);
+    const data = getFieldsByTable(state, props);
     return {
-        section: getSection(state, props),
+        table: getTable(state, props),
         entities: data,
         foreignKeys: getForeignKeys(state, props),
         loading: getLoading(state, props),
         loadingError: getError(state, props),
         user: getUser(state, props),
         isEditing: getIsEditing(state, props),
-        hasRevisionHistory: getHasRevisionHistory(state, props),
         fields: fieldsToFormFields(data)
     };
 }
@@ -68,13 +72,12 @@ const validate = (values, props) => {
     form: 'fields',
     validate
 })
-export default class ReferenceEntityList extends Component {
+export default class FieldList extends Component {
     static propTypes = {
         style: PropTypes.object.isRequired,
         entities: PropTypes.object.isRequired,
         foreignKeys: PropTypes.object.isRequired,
         isEditing: PropTypes.bool,
-        hasRevisionHistory: PropTypes.bool,
         startEditing: PropTypes.func.isRequired,
         endEditing: PropTypes.func.isRequired,
         startLoading: PropTypes.func.isRequired,
@@ -84,7 +87,7 @@ export default class ReferenceEntityList extends Component {
         handleSubmit: PropTypes.func.isRequired,
         user: PropTypes.object.isRequired,
         fields: PropTypes.object.isRequired,
-        section: PropTypes.object.isRequired,
+        table: PropTypes.object.isRequired,
         loading: PropTypes.bool,
         loadingError: PropTypes.object,
         submitting: PropTypes.bool,
@@ -97,12 +100,11 @@ export default class ReferenceEntityList extends Component {
             entities,
             fields,
             foreignKeys,
-            section,
+            table,
             loadingError,
             loading,
             user,
             isEditing,
-            hasRevisionHistory,
             startEditing,
             endEditing,
             resetForm,
@@ -113,18 +115,24 @@ export default class ReferenceEntityList extends Component {
         return (
             <form style={style} className="full"
                 onSubmit={handleSubmit(async (formFields) =>
-                    await tryUpdateFields(formFields, this.props)
+                    await actions.rUpdateFields(this.props.entities, formFields, this.props)
                 )}
             >
                 { isEditing &&
                     <EditHeader
-                        hasRevisionHistory={hasRevisionHistory}
+                        hasRevisionHistory={false}
                         reinitializeForm={resetForm}
                         endEditing={endEditing}
                         submitting={submitting}
                     />
                 }
-                <ReferenceHeader section={section} user={user} isEditing={isEditing} startEditing={startEditing} />
+                <EditableReferenceHeader 
+                    headerIcon="table2"
+                    name={`Fields in ${table.display_name}`}
+                    user={user} 
+                    isEditing={isEditing} 
+                    startEditing={startEditing} 
+                />
                 <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
                 { () => Object.keys(entities).length > 0 ?
                     <div className="wrapper wrapper--trim">
@@ -148,7 +156,7 @@ export default class ReferenceEntityList extends Component {
                                         <Field
                                             field={entity}
                                             foreignKeys={foreignKeys}
-                                            url={`${section.id}/${entity.id}`}
+                                            url={`/reference/databases/${table.db_id}/tables/${table.id}/fields/${entity.id}`}
                                             icon={getIconForField(entity)}
                                             isEditing={isEditing}
                                             formField={fields[entity.id]}
@@ -159,25 +167,7 @@ export default class ReferenceEntityList extends Component {
                     </div>
                     :
                     <div className={S.empty}>
-                        { section.empty &&
-                            <EmptyState
-                                title={section.empty.title}
-                                message={user.is_superuser ?
-                                    section.empty.adminMessage || section.empty.message :
-                                    section.empty.message
-                                }
-                                icon={section.empty.icon}
-                                image={section.empty.image}
-                                action={user.is_superuser ?
-                                    section.empty.adminAction || section.empty.action :
-                                    section.empty.action
-                                }
-                                link={user.is_superuser ?
-                                    section.empty.adminLink || section.empty.link :
-                                    section.empty.link
-                                }
-                            />
-                        }
+                        <EmptyState {...emptyStateData} />
                     </div>
                 }
                 </LoadingAndErrorWrapper>
diff --git a/frontend/src/metabase/reference/databases/FieldListContainer.jsx b/frontend/src/metabase/reference/databases/FieldListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..30d9d97c727dff3714c6ef1889519358a3ebb561
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/FieldListContainer.jsx
@@ -0,0 +1,78 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import TableSidebar from './TableSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import FieldList from "metabase/reference/databases/FieldList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabase,
+    getTable,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    database: getDatabase(state, props),    
+    table: getTable(state, props),    
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class FieldListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        table: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabaseMetadata(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            database,
+            table,
+            isEditing
+        } = this.props;
+
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<TableSidebar database={database} table={table}/>}
+            >
+                <FieldList {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/FieldSidebar.jsx b/frontend/src/metabase/reference/databases/FieldSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..171ee257214ed2c526c9209ad9cf78fc9759ed7e
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/FieldSidebar.jsx
@@ -0,0 +1,47 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const FieldSidebar =({
+    database,
+    table,
+    field,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <ul>
+            <div className={S.breadcrumbs}>
+                <Breadcrumbs
+                    className="py4"
+                    crumbs={[[database.name, `/reference/databases/${database.id}`],
+                             [table.name,`/reference/databases/${database.id}/tables/${table.id}`],
+                             [field.name]]}
+                    inSidebar={true}
+                    placeholder="Data Reference"
+                />
+            </div>
+                <SidebarItem key={`/reference/databases/${database.id}/tables/${table.id}/fields/${field.id}`} 
+                             href={`/reference/databases/${database.id}/tables/${table.id}/fields/${field.id}`} 
+                             icon="document" 
+                             name="Details" />
+        </ul>
+    </div>
+
+FieldSidebar.propTypes = {
+    database:          PropTypes.object,
+    table:          PropTypes.object,
+    field:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(FieldSidebar);
+
diff --git a/frontend/src/metabase/reference/databases/TableDetail.jsx b/frontend/src/metabase/reference/databases/TableDetail.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..4af371e7455c1a6814ec1a75bc36959a10e98207
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/TableDetail.jsx
@@ -0,0 +1,223 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+import { push } from "react-router-redux";
+
+import S from "metabase/reference/Reference.css";
+
+import List from "metabase/components/List.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+import Detail from "metabase/reference/components/Detail.jsx";
+import UsefulQuestions from "metabase/reference/components/UsefulQuestions.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+import {
+    getTable,
+    getFields,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getHasSingleSchema,
+    getIsFormulaExpanded,
+    getForeignKeys
+} from "../selectors";
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+
+const interestingQuestions = (table) => {
+    return [
+        {
+            text: `Count of ${table.display_name}`,
+            icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
+            link: getQuestionUrl({
+                dbId: table.db_id,
+                tableId: table.id,
+                getCount: true
+            })
+        },
+        {
+            text: `See raw data for ${table.display_name}`,
+            icon: "table2",
+            link: getQuestionUrl({
+                dbId: table.db_id,
+                tableId: table.id,
+            })
+        }
+    ]
+}
+const mapStateToProps = (state, props) => {
+    const entity = getTable(state, props) || {};
+    const fields = getFields(state, props);
+
+    return {
+        entity,
+        table: getTable(state, props),
+        metadataFields: fields,
+        loading: getLoading(state, props),
+        // naming this 'error' will conflict with redux form
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        foreignKeys: getForeignKeys(state, props),
+        isEditing: getIsEditing(state, props),
+        hasSingleSchema: getHasSingleSchema(state, props),
+        isFormulaExpanded: getIsFormulaExpanded(state, props),
+    }
+};
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions,
+    onChangeLocation: push
+};
+
+const validate = (values, props) => {
+    return {};
+}
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'details',
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats'],
+    validate
+})
+export default class TableDetail extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entity: PropTypes.object.isRequired,
+        table: PropTypes.object,
+        user: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        resetForm: PropTypes.func.isRequired,
+        fields: PropTypes.object.isRequired,
+        hasSingleSchema: PropTypes.bool,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool,
+    };
+
+    render() {
+        const {
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats },
+            style,
+            entity,
+            table,
+            loadingError,
+            loading,
+            user,
+            isEditing,
+            startEditing,
+            endEditing,
+            hasSingleSchema,
+            handleSubmit,
+            resetForm,
+            submitting,
+        } = this.props;
+
+        const onSubmit = handleSubmit(async (fields) =>
+            await actions.rUpdateTableDetail(fields, this.props)
+        );
+
+        return (
+            <form style={style} className="full"
+                onSubmit={onSubmit}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={false}
+                        onSubmit={onSubmit}
+                        endEditing={endEditing}
+                        reinitializeForm={resetForm}
+                        submitting={submitting}
+                        revisionMessageFormField={revision_message}
+                    />
+                }
+                <EditableReferenceHeader
+                    entity={entity}
+                    table={table}
+                    type="table"
+                    headerIcon="table2"
+                    headerLink={getQuestionUrl({ dbId: entity.db_id, tableId: entity.id})}
+                    name="Details"
+                    user={user}
+                    isEditing={isEditing}
+                    hasSingleSchema={hasSingleSchema}
+                    hasDisplayName={true}
+                    startEditing={startEditing}
+                    displayNameFormField={display_name}
+                    nameFormField={name}
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () =>
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            <li className="relative">
+                                <Detail
+                                    id="description"
+                                    name="Description"
+                                    description={entity.description}
+                                    placeholder="No description yet"
+                                    isEditing={isEditing}
+                                    field={description}
+                                />
+                            </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <Detail
+                                        id="name"
+                                        name="Actual name in database"
+                                        description={entity.name}
+                                        subtitleClass={S.tableActualName}
+                                    />
+                                </li>
+                            }
+                            <li className="relative">
+                                <Detail
+                                    id="points_of_interest"
+                                    name={`Why this table is interesting`}
+                                    description={entity.points_of_interest}
+                                    placeholder="Nothing interesting yet"
+                                    isEditing={isEditing}
+                                    field={points_of_interest}
+                                    />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="caveats"
+                                    name={`Things to be aware of about this table`}
+                                    description={entity.caveats}
+                                    placeholder="Nothing to be aware of yet"
+                                    isEditing={isEditing}
+                                    field={caveats}
+                                />
+                            </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <UsefulQuestions questions={interestingQuestions(this.props.table)} />
+                                </li>
+                            }
+                        </List>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/TableDetailContainer.jsx b/frontend/src/metabase/reference/databases/TableDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..e20d0c95c0de80e030f16d3fd4c1cccfc19a9ed3
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/TableDetailContainer.jsx
@@ -0,0 +1,78 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import TableSidebar from './TableSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import TableDetail from "metabase/reference/databases/TableDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabase,
+    getTable,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    database: getDatabase(state, props),    
+    table: getTable(state, props),    
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class TableDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        table: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabaseMetadata(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            database,
+            table,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<TableSidebar database={database} table={table}/>}
+            >
+                <TableDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/containers/ReferenceEntityList.jsx b/frontend/src/metabase/reference/databases/TableList.jsx
similarity index 55%
rename from frontend/src/metabase/reference/containers/ReferenceEntityList.jsx
rename to frontend/src/metabase/reference/databases/TableList.jsx
index 3773c59cf77c34e39bcdaf1bda60ddd36d2ec68e..2fbe9254fc84e6142f766f909536fa2a9da559b0 100644
--- a/frontend/src/metabase/reference/containers/ReferenceEntityList.jsx
+++ b/frontend/src/metabase/reference/databases/TableList.jsx
@@ -2,11 +2,8 @@
 import React, { Component } from "react";
 import PropTypes from "prop-types";
 import { connect } from "react-redux";
-import moment from "moment";
 
-import visualizations from "metabase/visualizations";
 import { isQueryable } from "metabase/lib/table";
-import * as Urls from "metabase/lib/urls";
 
 import S from "metabase/components/List.css";
 import R from "metabase/reference/Reference.css";
@@ -15,18 +12,14 @@ import List from "metabase/components/List.jsx";
 import ListItem from "metabase/components/ListItem.jsx";
 import EmptyState from "metabase/components/EmptyState.jsx";
 
+
 import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
 
 import ReferenceHeader from "../components/ReferenceHeader.jsx";
 
 import {
-    separateTablesBySchema
-} from '../utils';
-
-import {
-    getSection,
-    getData,
-    getUser,
+    getDatabase,
+    getTablesByDatabase,
     getHasSingleSchema,
     getError,
     getLoading
@@ -34,10 +27,15 @@ import {
 
 import * as metadataActions from "metabase/redux/metadata";
 
+
+const emptyStateData = {
+    message: `Tables in this database will appear here as they're added`,
+    icon: "table2"
+}
+
 const mapStateToProps = (state, props) => ({
-    section: getSection(state, props),
-    entities: getData(state, props),
-    user: getUser(state, props),
+    database: getDatabase(state, props),
+    entities: getTablesByDatabase(state, props),
     hasSingleSchema: getHasSingleSchema(state, props),
     loading: getLoading(state, props),
     loadingError: getError(state, props)
@@ -47,37 +45,53 @@ const mapDispatchToProps = {
     ...metadataActions
 };
 
-const createListItem = (entity, index, section) =>
+const createListItem = (entity, index) =>
     <li className="relative" key={entity.id}>
         <ListItem
             id={entity.id}
             index={index}
             name={entity.display_name || entity.name}
-            description={section.type !== 'questions' ?
-                entity.description :
-                `Created ${moment(entity.created_at).fromNow()} by ${entity.creator.common_name}`
-            }
-            url={section.type !== 'questions' ?
-                `${section.id}/${entity.id}` :
-                Urls.question(entity.id)
-            }
-            icon={section.type === 'questions' ?
-                visualizations.get(entity.display).iconName :
-                section.icon
-            }
+            description={ entity.description }
+            url={ `/reference/databases/${entity.db_id}/tables/${entity.id}` }
+            icon="table2"
         />
     </li>;
 
+
 const createSchemaSeparator = (entity) =>
     <li className={R.schemaSeparator}>{entity.schema}</li>;
 
+
+export const separateTablesBySchema = (
+    tables,
+    createSchemaSeparator,
+    createListItem
+) => Object.values(tables)
+    .sort((table1, table2) => table1.schema > table2.schema ? 1 :
+        table1.schema === table2.schema ? 0 : -1
+    )
+    .map((table, index, sortedTables) => {
+        if (!table || !table.id || !table.name) {
+            return;
+        }
+        // add schema header for first element and if schema is different from previous
+        const previousTableId = Object.keys(sortedTables)[index - 1];
+        return index === 0 ||
+            sortedTables[previousTableId].schema !== table.schema ?
+                [
+                    createSchemaSeparator(table),
+                    createListItem(table, index)
+                ] :
+                createListItem(table, index);
+    });
+
+
 @connect(mapStateToProps, mapDispatchToProps)
-export default class ReferenceEntityList extends Component {
+export default class TableList extends Component {
     static propTypes = {
         style: PropTypes.object.isRequired,
         entities: PropTypes.object.isRequired,
-        user: PropTypes.object.isRequired,
-        section: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
         hasSingleSchema: PropTypes.bool,
         loading: PropTypes.bool,
         loadingError: PropTypes.object
@@ -86,9 +100,8 @@ export default class ReferenceEntityList extends Component {
     render() {
         const {
             entities,
-            user,
             style,
-            section,
+            database,
             hasSingleSchema,
             loadingError,
             loading
@@ -96,46 +109,31 @@ export default class ReferenceEntityList extends Component {
 
         return (
             <div style={style} className="full">
-                <ReferenceHeader section={section} />
+                <ReferenceHeader 
+                    name={`Tables in ${database.name}`}
+                    type="tables"
+                    headerIcon="database"
+                />
                 <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
                 { () => Object.keys(entities).length > 0 ?
                     <div className="wrapper wrapper--trim">
                         <List>
-                            { section.type === "tables" && !hasSingleSchema ?
+                            { !hasSingleSchema ?
                                 separateTablesBySchema(
                                     entities,
-                                    section,
                                     createSchemaSeparator,
                                     createListItem
                                 ) :
                                 Object.values(entities).filter(isQueryable).map((entity, index) =>
                                     entity && entity.id && entity.name &&
-                                        createListItem(entity, index, section)
+                                        createListItem(entity, index)
                                 )
                             }
                         </List>
                     </div>
                     :
                     <div className={S.empty}>
-                        { section.empty &&
-                            <EmptyState
-                                title={section.empty.title}
-                                message={user.is_superuser ?
-                                    section.empty.adminMessage || section.empty.message :
-                                    section.empty.message
-                                }
-                                icon={section.empty.icon}
-                                image={section.empty.image}
-                                action={user.is_superuser ?
-                                    section.empty.adminAction || section.empty.action :
-                                    section.empty.action
-                                }
-                                link={user.is_superuser ?
-                                    section.empty.adminLink || section.empty.link :
-                                    section.empty.link
-                                }
-                            />
-                        }
+                        <EmptyState {...emptyStateData}/>
                     </div>
                 }
                 </LoadingAndErrorWrapper>
diff --git a/frontend/src/metabase/reference/databases/TableListContainer.jsx b/frontend/src/metabase/reference/databases/TableListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..7eb48ef50aed613061f6100c3c0199b1508c8c58
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/TableListContainer.jsx
@@ -0,0 +1,74 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import DatabaseSidebar from './DatabaseSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import TableList from "metabase/reference/databases/TableList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabase,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    database: getDatabase(state, props),    
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class TableListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchDatabaseMetadata(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+
+    }
+
+    render() {
+        const {
+            database,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<DatabaseSidebar database={database} />}
+            >
+                <TableList {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/databases/TableQuestions.jsx b/frontend/src/metabase/reference/databases/TableQuestions.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..fe94f2292a9005ee42012c3c1dd084b56087031e
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/TableQuestions.jsx
@@ -0,0 +1,115 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import moment from "moment";
+
+import visualizations from "metabase/visualizations";
+import { isQueryable } from "metabase/lib/table";
+import * as Urls from "metabase/lib/urls";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+
+import {
+    getTableQuestions,
+    getError,
+    getLoading,
+    getTable
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+const emptyStateData = (table) =>  {
+    return {
+        message: "Questions about this table will appear here as they're added",
+        icon: "all",
+        action: "Ask a question",
+        link: getQuestionUrl({
+            dbId: table.db_id,
+            tableId: table.id,
+        })
+    }
+}
+
+
+const mapStateToProps = (state, props) => ({
+    table: getTable(state, props),
+    entities: getTableQuestions(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class TableQuestions extends Component {
+    static propTypes = {
+        table: PropTypes.object.isRequired,
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name={`Questions about ${this.props.table.display_name}`}
+                    type="questions"
+                    headerIcon="table2"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            { 
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                            <li className="relative" key={entity.id}>
+                                                <ListItem
+                                                    id={entity.id}
+                                                    index={index}
+                                                    name={entity.display_name || entity.name}
+                                                    description={ `Created ${moment(entity.created_at).fromNow()} by ${entity.creator.common_name}` }
+                                                    url={ Urls.question(entity.id) }
+                                                    icon={ visualizations.get(entity.display).iconName }
+                                                />
+                                            </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData(this.props.table)}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/containers/ReferenceApp.jsx b/frontend/src/metabase/reference/databases/TableQuestionsContainer.jsx
similarity index 50%
rename from frontend/src/metabase/reference/containers/ReferenceApp.jsx
rename to frontend/src/metabase/reference/databases/TableQuestionsContainer.jsx
index de4bec50e79ccc36883d2c535ba6366c50e8aac8..33f9893606133562223507b65a8aee7dc2fcff4f 100644
--- a/frontend/src/metabase/reference/containers/ReferenceApp.jsx
+++ b/frontend/src/metabase/reference/databases/TableQuestionsContainer.jsx
@@ -3,98 +3,79 @@ import React, { Component } from 'react';
 import PropTypes from "prop-types";
 import { connect } from 'react-redux';
 
-import Sidebar from 'metabase/components/Sidebar.jsx';
+import TableSidebar from './TableSidebar.jsx';
 import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
 
+import TableQuestions from "metabase/reference/databases/TableQuestions.jsx"
 import * as metadataActions from 'metabase/redux/metadata';
 import * as actions from 'metabase/reference/reference';
 
 import {
+    getDatabase,
+    getTable,
     getDatabaseId,
-    getSectionId,
-    getSections,
-    getSection,
-    getBreadcrumbs,
     getIsEditing
 } from '../selectors';
 
-import {
-    tryFetchData
-} from '../utils';
-
 import {
     loadEntities
 } from 'metabase/questions/questions';
 
-import {
-    fetchDashboards
-} from 'metabase/dashboards/dashboards';
 
 const mapStateToProps = (state, props) => ({
-    sectionId: getSectionId(state, props),
+    database: getDatabase(state, props),    
+    table: getTable(state, props),    
     databaseId: getDatabaseId(state, props),
-    sections: getSections(state, props),
-    section: getSection(state, props),
-    breadcrumbs: getBreadcrumbs(state, props),
     isEditing: getIsEditing(state, props)
 });
 
 const mapDispatchToProps = {
-    fetchQuestions: () => loadEntities("card", {}),
-    fetchDashboards,
+    fetchQuestions: () => loadEntities("cards", {}),
     ...metadataActions,
     ...actions
 };
 
 @connect(mapStateToProps, mapDispatchToProps)
-export default class ReferenceApp extends Component {
+export default class TableQuestionsContainer extends Component {
     static propTypes = {
         params: PropTypes.object.isRequired,
-        breadcrumbs: PropTypes.array,
         location: PropTypes.object.isRequired,
-        children: PropTypes.any.isRequired,
-        sections: PropTypes.object.isRequired,
-        section: PropTypes.object.isRequired,
+        database: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        table: PropTypes.object.isRequired,
         isEditing: PropTypes.bool
     };
 
-    async componentWillMount() {
-        await tryFetchData(this.props);
+    async fetchContainerData() {
+        await actions.wrappedFetchDatabaseMetadataAndQuestion(this.props, this.props.databaseId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
     }
 
-    async componentWillReceiveProps(newProps) {
+    componentWillReceiveProps(newProps) {
         if (this.props.location.pathname === newProps.location.pathname) {
             return;
         }
 
-        newProps.endEditing();
-        newProps.endLoading();
-        newProps.clearError();
-        newProps.collapseFormula();
-
-        await tryFetchData(newProps);
+        actions.clearState(newProps)
     }
 
     render() {
         const {
-            children,
-            section,
-            sections,
-            breadcrumbs,
+            database,
+            table,
             isEditing
         } = this.props;
 
-        if (section.sidebar === false) {
-            return children;
-        }
-
         return (
             <SidebarLayout
                 className="flex-full relative"
                 style={ isEditing ? { paddingTop: '43px' } : {}}
-                sidebar={<Sidebar sections={sections} breadcrumbs={breadcrumbs} />}
+                sidebar={<TableSidebar database={database} table={table}/>}
             >
-                {children}
+                <TableQuestions {...this.props} />
             </SidebarLayout>
         );
     }
diff --git a/frontend/src/metabase/reference/databases/TableSidebar.jsx b/frontend/src/metabase/reference/databases/TableSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..c221a3d3ed06c4b1b282764e05ffbd60b09c9650
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/TableSidebar.jsx
@@ -0,0 +1,53 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const TableSidebar = ({
+    database,
+    table,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <div className={S.breadcrumbs}>
+            <Breadcrumbs
+                className="py4"
+                crumbs={[["Databases","/reference/databases"],
+                         [database.name, `/reference/databases/${database.id}`],
+                         [table.name]]}
+                inSidebar={true}
+                placeholder="Data Reference"
+            />
+        </div>
+        <ol>
+            <SidebarItem key={`/reference/databases/${database.id}/tables/${table.id}`} 
+                         href={`/reference/databases/${database.id}/tables/${table.id}`} 
+                         icon="document" 
+                         name="Details" />
+            <SidebarItem key={`/reference/databases/${database.id}/tables/${table.id}/fields`} 
+                         href={`/reference/databases/${database.id}/tables/${table.id}/fields`} 
+                         icon="fields" 
+                         name="Fields in this table" />
+            <SidebarItem key={`/reference/databases/${database.id}/tables/${table.id}/questions`} 
+                         href={`/reference/databases/${database.id}/tables/${table.id}/questions`} 
+                         icon="all" 
+                         name="Questions about this table" />
+        </ol>
+    </div>
+
+TableSidebar.propTypes = {
+    database:          PropTypes.object,
+    table:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(TableSidebar);
+
diff --git a/frontend/src/metabase/reference/databases/databases.integ.spec.js b/frontend/src/metabase/reference/databases/databases.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..6e49c1db58e129b8a167cf7c09eb871756dd7785
--- /dev/null
+++ b/frontend/src/metabase/reference/databases/databases.integ.spec.js
@@ -0,0 +1,144 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { CardApi } from 'metabase/services'
+
+import { 
+    FETCH_DATABASE_METADATA,
+    FETCH_DATABASES
+} from "metabase/redux/metadata";
+
+import { END_LOADING } from "metabase/reference/reference"
+
+import DatabaseListContainer from "metabase/reference/databases/DatabaseListContainer";
+import DatabaseDetailContainer from "metabase/reference/databases/DatabaseDetailContainer";
+import TableListContainer from "metabase/reference/databases/TableListContainer";
+import TableDetailContainer from "metabase/reference/databases/TableDetailContainer";
+import TableQuestionsContainer from "metabase/reference/databases/TableQuestionsContainer";
+import FieldListContainer from "metabase/reference/databases/FieldListContainer";
+import FieldDetailContainer from "metabase/reference/databases/FieldDetailContainer";
+
+import DatabaseList from "metabase/reference/databases/DatabaseList";
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+describe("The Reference Section", () => {
+    // Test data
+    const cardDef = { name :"A card", display: "scalar", 
+                      dataset_query: {database: 1, table_id: 1, type: "query", query: {source_table: 1, "aggregation": ["count"]}},
+                      visualization_settings: {}}
+
+    // Scaffolding
+    beforeAll(async () => {
+        await login();
+
+    })
+
+    describe("The Data Reference for the Sample Database", async () => {
+        
+        // database list
+        it("should see databases", async () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/");
+            var container = mount(store.connectContainer(<DatabaseListContainer />));
+            await store.waitForActions([FETCH_DATABASES, END_LOADING])
+            
+            expect(container.find(ReferenceHeader).length).toBe(1)
+            expect(container.find(DatabaseList).length).toBe(1)            
+            expect(container.find(AdminAwareEmptyState).length).toBe(0)
+            
+            expect(container.find(List).length).toBe(1)
+            expect(container.find(ListItem).length).toBeGreaterThanOrEqual(1)
+        })
+        
+        // database detail
+        it("should see a the detail view for the sample database", async ()=>{
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1");
+            mount(store.connectContainer(<DatabaseDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+
+        })
+        
+        // table list
+       it("should see the 4 tables in the sample database",async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables");
+            mount(store.connectContainer(<TableListContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+        // table detail
+
+       it("should see the Orders table", async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/1");
+            mount(store.connectContainer(<TableDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+
+       it("should see the Reviews table", async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/2");
+            mount(store.connectContainer(<TableDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+       it("should see the Products table", async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/3");
+            mount(store.connectContainer(<TableDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+       it("should see the People table", async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/4");
+            mount(store.connectContainer(<TableDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+        // field list
+       it("should see the fields for the orders table", async  () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/1/fields");
+            mount(store.connectContainer(<FieldListContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+
+        })
+       it("should see the questions for the orders tables", async  () => {
+
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/1/questions");
+            mount(store.connectContainer(<TableQuestionsContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+            
+            var card = await CardApi.create(cardDef)
+
+            expect(card.name).toBe(cardDef.name);
+            
+            await CardApi.delete({cardId: card.id})
+        })
+
+        // field detail
+
+       it("should see the orders created_at timestamp field", async () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/1/fields/1");
+            mount(store.connectContainer(<FieldDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+
+       it("should see the orders id field", async () => {
+            const store = await createTestStore()
+            store.pushPath("/reference/databases/1/tables/1/fields/25");
+            mount(store.connectContainer(<FieldDetailContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA])
+        })
+    });
+
+
+});
\ No newline at end of file
diff --git a/frontend/src/metabase/reference/guide/BaseSidebar.jsx b/frontend/src/metabase/reference/guide/BaseSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..e52cd9f9ac9d0533f6e2b494812b173e0d6ffbb0
--- /dev/null
+++ b/frontend/src/metabase/reference/guide/BaseSidebar.jsx
@@ -0,0 +1,51 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const BaseSidebar = ({
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <div className={S.breadcrumbs}>
+            <Breadcrumbs
+                className="py4"
+                crumbs={[["Data Reference"]]}
+                inSidebar={true}
+                placeholder="Data Reference"
+            />
+        </div>
+        <ol>
+            <SidebarItem key="/reference/guide" 
+                         href="/reference/guide" 
+                         icon="reference" 
+                         name="Start here" />
+            <SidebarItem key="/reference/metrics" 
+                         href="/reference/metrics" 
+                         icon="ruler" 
+                         name="Metrics" />
+            <SidebarItem key="/reference/segments" 
+                         href="/reference/segments" 
+                         icon="segment" 
+                         name="Segments" />
+            <SidebarItem key="/reference/databases" 
+                         href="/reference/databases" 
+                         icon="database" 
+                         name="Databases and tables" />
+        </ol>
+    </div>
+
+BaseSidebar.propTypes = {
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(BaseSidebar);
+
diff --git a/frontend/src/metabase/reference/guide/GettingStartedGuideContainer.jsx b/frontend/src/metabase/reference/guide/GettingStartedGuideContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..54f40a982b0296c7426ad83f4cd3c0ce5cd31ca1
--- /dev/null
+++ b/frontend/src/metabase/reference/guide/GettingStartedGuideContainer.jsx
@@ -0,0 +1,63 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import ReferenceGettingStartedGuide from "metabase/reference/guide/ReferenceGettingStartedGuide.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+import {
+    fetchDashboards
+} from 'metabase/dashboards/dashboards';
+
+const mapStateToProps = (state, props) => ({
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    fetchDashboards,
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class GettingStartedGuideContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData() {
+        await actions.wrappedFetchGuide(this.props);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+
+        return (
+                <ReferenceGettingStartedGuide {...this.props} />
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/containers/ReferenceGettingStartedGuide.jsx b/frontend/src/metabase/reference/guide/ReferenceGettingStartedGuide.jsx
similarity index 98%
rename from frontend/src/metabase/reference/containers/ReferenceGettingStartedGuide.jsx
rename to frontend/src/metabase/reference/guide/ReferenceGettingStartedGuide.jsx
index d0a637dcb108746d81f88279a5188131d8cfd869..e572ebb59b3935ee9ebec44a617e0c316a9b5c89 100644
--- a/frontend/src/metabase/reference/containers/ReferenceGettingStartedGuide.jsx
+++ b/frontend/src/metabase/reference/guide/ReferenceGettingStartedGuide.jsx
@@ -45,11 +45,25 @@ import {
 
 import {
     getQuestionUrl,
-    has,
-    isGuideEmpty,
-    tryUpdateGuide
+    has
 } from '../utils';
 
+const isGuideEmpty = ({
+    things_to_know,
+    contact,
+    most_important_dashboard,
+    important_metrics,
+    important_segments,
+    important_tables
+} = {}) => things_to_know ? false :
+    contact && contact.name ? false :
+    contact && contact.email ? false :
+    most_important_dashboard ? false :
+    important_metrics && important_metrics.length !== 0 ? false :
+    important_segments && important_segments.length !== 0 ? false :
+    important_tables && important_tables.length !== 0 ? false :
+    true;
+
 const mapStateToProps = (state, props) => {
     const guide = getGuide(state, props);
     const dashboards = getDashboards(state, props);
@@ -200,7 +214,7 @@ export default class ReferenceGettingStartedGuide extends Component {
         } = this.props;
 
         const onSubmit = handleSubmit(async (fields) =>
-            await tryUpdateGuide(fields, this.props)
+            await actions.tryUpdateGuide(fields, this.props)
         );
 
         const getSelectedIds = fields => fields
diff --git a/frontend/src/metabase/reference/guide/guide.integ.spec.js b/frontend/src/metabase/reference/guide/guide.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..59896027a17ec04a73c600349aa0db8fc2e893e5
--- /dev/null
+++ b/frontend/src/metabase/reference/guide/guide.integ.spec.js
@@ -0,0 +1,86 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { SegmentApi, MetricApi } from 'metabase/services'
+
+import { 
+    FETCH_DATABASE_METADATA,
+    FETCH_METRICS,
+    FETCH_SEGMENTS
+} from "metabase/redux/metadata";
+
+import GettingStartedGuideContainer from "metabase/reference/guide/GettingStartedGuideContainer";
+
+
+
+describe("The Reference Section", () => {
+    // Test data
+    const segmentDef = {name: "A Segment", description: "I did it!", table_id: 1, show_in_getting_started: true,
+                        definition: {database: 1, query: {filter: ["abc"]}}}
+
+    const anotherSegmentDef = {name: "Another Segment", description: "I did it again!", table_id: 1, show_in_getting_started: true,
+                               definition:{database: 1, query: {filter: ["def"]}}}
+    const metricDef = {name: "A Metric", description: "I did it!", table_id: 1,show_in_getting_started: true,
+                        definition: {database: 1, query: {aggregation: ["count"]}}}
+
+    const anotherMetricDef = {name: "Another Metric", description: "I did it again!", table_id: 1,show_in_getting_started: true,
+                        definition: {database: 1, query: {aggregation: ["count"]}}}
+    
+    // Scaffolding
+    beforeAll(async () => {
+        await login();
+
+    })
+
+
+    describe("The Getting Started Guide", async ()=>{
+        
+        
+        it("Should show an empty guide for non-admin users", async () => {
+            const store = await createTestStore()    
+            store.pushPath("/reference/");
+            mount(store.connectContainer(<GettingStartedGuideContainer />));
+            await store.waitForActions([FETCH_DATABASE_METADATA, FETCH_SEGMENTS, FETCH_METRICS])
+        })
+        
+        xit("Should show an empty guide with a creation CTA for admin users", async () => {
+        })
+
+        xit("A non-admin attempting to edit the guide should get an error", async () => {
+        })
+
+        it("Adding metrics should to the guide should make them appear", async () => {
+            
+            expect(0).toBe(0)
+            var metric = await MetricApi.create(metricDef);
+            expect(1).toBe(1)
+            var metric2 = await MetricApi.create(anotherMetricDef);
+            expect(2).toBe(2)
+            await MetricApi.delete({metricId: metric.id, revision_message: "Please"})
+            expect(1).toBe(1)
+            await MetricApi.delete({metricId: metric2.id, revision_message: "Please"})
+            expect(0).toBe(0)
+        })
+
+        it("Adding segments should to the guide should make them appear", async () => {
+            expect(0).toBe(0)
+            var segment = await SegmentApi.create(segmentDef);
+            expect(1).toBe(1)
+            var anotherSegment = await SegmentApi.create(anotherSegmentDef);
+            expect(2).toBe(2)
+            await SegmentApi.delete({segmentId: segment.id, revision_message: "Please"})
+            expect(1).toBe(1)
+            await SegmentApi.delete({segmentId: anotherSegment.id, revision_message: "Please"})
+            expect(0).toBe(0)
+        })
+        
+    })
+
+
+
+});
\ No newline at end of file
diff --git a/frontend/src/metabase/reference/containers/ReferenceEntity.jsx b/frontend/src/metabase/reference/metrics/MetricDetail.jsx
similarity index 57%
rename from frontend/src/metabase/reference/containers/ReferenceEntity.jsx
rename to frontend/src/metabase/reference/metrics/MetricDetail.jsx
index 45a0b6ec4ac7f76d968d2fa79cae07b745639607..e72f971cf43086e2e72a5682a91dac6756b7c4fd 100644
--- a/frontend/src/metabase/reference/containers/ReferenceEntity.jsx
+++ b/frontend/src/metabase/reference/metrics/MetricDetail.jsx
@@ -5,38 +5,29 @@ import { connect } from "react-redux";
 import { reduxForm } from "redux-form";
 import { push } from "react-router-redux";
 
-import S from "metabase/reference/Reference.css";
-
 import List from "metabase/components/List.jsx";
 import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
 
 import EditHeader from "metabase/reference/components/EditHeader.jsx";
-import ReferenceHeader from "metabase/reference/components/ReferenceHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
 import Detail from "metabase/reference/components/Detail.jsx";
-import FieldTypeDetail from "metabase/reference/components/FieldTypeDetail.jsx";
-import UsefulQuestions from "metabase/reference/components/UsefulQuestions.jsx";
 import FieldsToGroupBy from "metabase/reference/components/FieldsToGroupBy.jsx";
 import Formula from "metabase/reference/components/Formula.jsx";
 import MetricImportantFieldsDetail from "metabase/reference/components/MetricImportantFieldsDetail.jsx";
 
 import {
-    tryUpdateData
+    getQuestionUrl
 } from '../utils';
 
 import {
-    getSection,
-    getData,
+    getMetric,
     getTable,
     getFields,
     getGuide,
     getError,
     getLoading,
     getUser,
-    getHasQuestions,
     getIsEditing,
-    getHasDisplayName,
-    getHasRevisionHistory,
-    getHasSingleSchema,
     getIsFormulaExpanded,
     getForeignKeys
 } from "../selectors";
@@ -44,8 +35,9 @@ import {
 import * as metadataActions from 'metabase/redux/metadata';
 import * as actions from 'metabase/reference/reference';
 
+
 const mapStateToProps = (state, props) => {
-    const entity = getData(state, props) || {};
+    const entity = getMetric(state, props) || {};
     const guide = getGuide(state, props);
     const fields = getFields(state, props);
 
@@ -58,7 +50,6 @@ const mapStateToProps = (state, props) => {
     };
 
     return {
-        section: getSection(state, props),
         entity,
         table: getTable(state, props),
         metadataFields: fields,
@@ -69,11 +60,7 @@ const mapStateToProps = (state, props) => {
         user: getUser(state, props),
         foreignKeys: getForeignKeys(state, props),
         isEditing: getIsEditing(state, props),
-        hasSingleSchema: getHasSingleSchema(state, props),
-        hasQuestions: getHasQuestions(state, props),
-        hasDisplayName: getHasDisplayName(state, props),
         isFormulaExpanded: getIsFormulaExpanded(state, props),
-        hasRevisionHistory: getHasRevisionHistory(state, props),
         initialValues,
     }
 };
@@ -84,18 +71,16 @@ const mapDispatchToProps = {
     onChangeLocation: push
 };
 
-const validate = (values, props) => props.hasRevisionHistory ?
-    !values.revision_message ?
-        { revision_message: "Please enter a revision message" } : {} :
-    {};
+const validate = (values, props) =>  !values.revision_message ? 
+    { revision_message: "Please enter a revision message" } : {} 
 
 @connect(mapStateToProps, mapDispatchToProps)
 @reduxForm({
     form: 'details',
-    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats', 'how_is_this_calculated', 'special_type', 'fk_target_field_id', 'important_fields'],
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats', 'how_is_this_calculated', 'important_fields'],
     validate
 })
-export default class ReferenceEntity extends Component {
+export default class MetricDetail extends Component {
     static propTypes = {
         style: PropTypes.object.isRequired,
         entity: PropTypes.object.isRequired,
@@ -103,9 +88,7 @@ export default class ReferenceEntity extends Component {
         metadataFields: PropTypes.object,
         guide: PropTypes.object,
         user: PropTypes.object.isRequired,
-        foreignKeys: PropTypes.object,
         isEditing: PropTypes.bool,
-        hasQuestions: PropTypes.bool,
         startEditing: PropTypes.func.isRequired,
         endEditing: PropTypes.func.isRequired,
         startLoading: PropTypes.func.isRequired,
@@ -117,11 +100,7 @@ export default class ReferenceEntity extends Component {
         handleSubmit: PropTypes.func.isRequired,
         resetForm: PropTypes.func.isRequired,
         fields: PropTypes.object.isRequired,
-        section: PropTypes.object.isRequired,
-        hasSingleSchema: PropTypes.bool,
-        hasDisplayName: PropTypes.bool,
         isFormulaExpanded: PropTypes.bool,
-        hasRevisionHistory: PropTypes.bool,
         loading: PropTypes.bool,
         loadingError: PropTypes.object,
         submitting: PropTypes.bool,
@@ -130,9 +109,8 @@ export default class ReferenceEntity extends Component {
 
     render() {
         const {
-            fields: { name, display_name, description, revision_message, points_of_interest, caveats, how_is_this_calculated, special_type, fk_target_field_id, important_fields },
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats, how_is_this_calculated, important_fields },
             style,
-            section,
             entity,
             table,
             metadataFields,
@@ -140,17 +118,12 @@ export default class ReferenceEntity extends Component {
             loadingError,
             loading,
             user,
-            foreignKeys,
             isEditing,
-            hasQuestions,
             startEditing,
             endEditing,
             expandFormula,
             collapseFormula,
-            hasSingleSchema,
-            hasDisplayName,
             isFormulaExpanded,
-            hasRevisionHistory,
             handleSubmit,
             resetForm,
             submitting,
@@ -158,7 +131,7 @@ export default class ReferenceEntity extends Component {
         } = this.props;
 
         const onSubmit = handleSubmit(async (fields) =>
-            await tryUpdateData(fields, this.props)
+            await actions.rUpdateMetricDetail(this.props.entity, this.props.guide, fields, this.props)
         );
 
         return (
@@ -167,7 +140,7 @@ export default class ReferenceEntity extends Component {
             >
                 { isEditing &&
                     <EditHeader
-                        hasRevisionHistory={hasRevisionHistory}
+                        hasRevisionHistory={true}
                         onSubmit={onSubmit}
                         endEditing={endEditing}
                         reinitializeForm={resetForm}
@@ -175,14 +148,17 @@ export default class ReferenceEntity extends Component {
                         revisionMessageFormField={revision_message}
                     />
                 }
-                <ReferenceHeader
+                <EditableReferenceHeader
                     entity={entity}
                     table={table}
-                    section={section}
+                    type="metric"
+                    headerIcon="ruler"
+                    headerLink={getQuestionUrl({ dbId: table && table.db_id, tableId: entity.table_id, metricId: entity.id})}
+                    name="Details"
                     user={user}
                     isEditing={isEditing}
-                    hasSingleSchema={hasSingleSchema}
-                    hasDisplayName={hasDisplayName}
+                    hasSingleSchema={false}
+                    hasDisplayName={false}
                     startEditing={startEditing}
                     displayNameFormField={display_name}
                     nameFormField={name}
@@ -201,20 +177,10 @@ export default class ReferenceEntity extends Component {
                                     field={description}
                                 />
                             </li>
-                            { hasDisplayName && !isEditing &&
-                                <li className="relative">
-                                    <Detail
-                                        id="name"
-                                        name="Actual name in database"
-                                        description={entity.name}
-                                        subtitleClass={S.tableActualName}
-                                    />
-                                </li>
-                            }
                             <li className="relative">
                                 <Detail
                                     id="points_of_interest"
-                                    name={`Why this ${section.type} is interesting`}
+                                    name="Why this Metric is interesting"
                                     description={entity.points_of_interest}
                                     placeholder="Nothing interesting yet"
                                     isEditing={isEditing}
@@ -224,81 +190,50 @@ export default class ReferenceEntity extends Component {
                             <li className="relative">
                                 <Detail
                                     id="caveats"
-                                    name={`Things to be aware of about this ${section.type}`}
+                                    name="Things to be aware of about this Metric"
                                     description={entity.caveats}
                                     placeholder="Nothing to be aware of yet"
                                     isEditing={isEditing}
                                     field={caveats}
                                 />
                             </li>
-                            { section.type === 'metric' &&
-                                <li className="relative">
-                                    <Detail
-                                        id="how_is_this_calculated"
-                                        name={`How this ${section.type} is calculated`}
-                                        description={entity.how_is_this_calculated}
-                                        placeholder="Nothing on how it's calculated yet"
-                                        isEditing={isEditing}
-                                        field={how_is_this_calculated}
-                                    />
-                                </li>
-                            }
-                            { (section.type === 'metric' || section.type === 'segment') &&
-                                table && !isEditing &&
+                            <li className="relative">
+                                <Detail
+                                    id="how_is_this_calculated"
+                                    name="How this Metric is calculated"
+                                    description={entity.how_is_this_calculated}
+                                    placeholder="Nothing on how it's calculated yet"
+                                    isEditing={isEditing}
+                                    field={how_is_this_calculated}
+                                />
+                            </li>
+                            {   table && !isEditing &&
                                 <li className="relative">
                                     <Formula
-                                        type={section.type}
+                                        type="metric"
                                         entity={entity}
-                                        table={table}
                                         isExpanded={isFormulaExpanded}
                                         expandFormula={expandFormula}
                                         collapseFormula={collapseFormula}
                                     />
                                 </li>
                             }
-                            { !isEditing && section.type === 'field' &&
-                                <li className="relative">
-                                    <Detail
-                                        id="base_type"
-                                        name={`Data type`}
-                                        description={entity.base_type}
-                                    />
-                                </li>
-                            }
-                            { section.type === 'field' &&
-                                <li className="relative">
-                                    <FieldTypeDetail
-                                        field={entity}
-                                        foreignKeys={foreignKeys}
-                                        fieldTypeFormField={special_type}
-                                        foreignKeyFormField={fk_target_field_id}
-                                        isEditing={isEditing}
-                                    />
-                                </li>
-                            }
-                            { hasQuestions && !isEditing &&
-                                <li className="relative">
-                                    <UsefulQuestions questions={section.questions} />
-                                </li>
-                            }
-                            { section.type === 'metric' &&
-                                <li className="relative">
-                                    <MetricImportantFieldsDetail
-                                        fields={guide && guide.metric_important_fields[entity.id] &&
-                                            Object.values(guide.metric_important_fields[entity.id])
-                                                .map(fieldId => metadataFields[fieldId])
-                                                .reduce((map, field) => ({ ...map, [field.id]: field }), {})
-                                        }
-                                        table={table}
-                                        allFields={metadataFields}
-                                        metric={entity}
-                                        onChangeLocation={onChangeLocation}
-                                        isEditing={isEditing}
-                                        formField={important_fields}
-                                    />
-                                </li>
-                            }
-                            { section.type === 'metric' && !isEditing &&
+                            <li className="relative">
+                                <MetricImportantFieldsDetail
+                                    fields={guide && guide.metric_important_fields[entity.id] &&
+                                        Object.values(guide.metric_important_fields[entity.id])
+                                            .map(fieldId => metadataFields[fieldId])
+                                            .reduce((map, field) => ({ ...map, [field.id]: field }), {})
+                                    }
+                                    table={table}
+                                    allFields={metadataFields}
+                                    metric={entity}
+                                    onChangeLocation={onChangeLocation}
+                                    isEditing={isEditing}
+                                    formField={important_fields}
+                                />
+                            </li>
+                            { !isEditing &&
                                 <li className="relative">
                                     <FieldsToGroupBy
                                         fields={table.fields
@@ -308,7 +243,7 @@ export default class ReferenceEntity extends Component {
                                             .map(fieldId => metadataFields[fieldId])
                                             .reduce((map, field) => ({ ...map, [field.id]: field }), {})
                                         }
-                                        databaseId={table.db_id}
+                                        databaseId={table && table.db_id}
                                         metric={entity}
                                         title={ guide && guide.metric_important_fields[entity.id] ?
                                             "Other fields you can group this metric by" :
diff --git a/frontend/src/metabase/reference/metrics/MetricDetailContainer.jsx b/frontend/src/metabase/reference/metrics/MetricDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..3592e969be44b7d221e31e72559ad93b2ed06ef7
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricDetailContainer.jsx
@@ -0,0 +1,80 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import MetricSidebar from './MetricSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import MetricDetail from "metabase/reference/metrics/MetricDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getMetric,
+    getMetricId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    metric: getMetric(state, props),
+    metricId: getMetricId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        user: PropTypes.object.isRequired,
+        metric: PropTypes.object.isRequired,
+        metricId: PropTypes.number.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchMetricDetail(this.props, this.props.metricId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            isEditing,
+            user,
+            metric
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<MetricSidebar metric={metric} user={user}/>}
+            >
+                <MetricDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/metrics/MetricList.jsx b/frontend/src/metabase/reference/metrics/MetricList.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..c19a6ecbcd3536de35eda2946b3f8dd6dbef13b3
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricList.jsx
@@ -0,0 +1,99 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+
+import { isQueryable } from "metabase/lib/table";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getMetrics,
+    getError,
+    getLoading
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+
+const emptyStateData = {
+    title: "Metrics are the official numbers that your team cares about",
+    adminMessage: "Defining common metrics for your team makes it even easier to ask questions",
+    message: "Metrics will appear here once your admins have created some",
+    image: "app/assets/img/metrics-list",
+    adminAction: "Learn how to create metrics",
+    adminLink: "http://www.metabase.com/docs/latest/administration-guide/06-segments-and-metrics.html"
+}
+
+const mapStateToProps = (state, props) => ({
+    entities: getMetrics(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricList extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name="Metrics"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            {
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                          <li className="relative" key={entity.id}>
+                                                <ListItem
+                                                    id={entity.id}
+                                                    index={index}
+                                                    name={entity.display_name || entity.name}
+                                                    description={ entity.description }
+                                                    url={ `/reference/metrics/${entity.id}` }
+                                                    icon="ruler"
+                                                />
+                                            </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/metrics/MetricListContainer.jsx b/frontend/src/metabase/reference/metrics/MetricListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..5a566c44d1402be76165bd625f088a6da3eecb8b
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricListContainer.jsx
@@ -0,0 +1,70 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import BaseSidebar from 'metabase/reference/guide/BaseSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import MetricList from "metabase/reference/metrics/MetricList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+const mapStateToProps = (state, props) => ({
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+
+    async fetchContainerData(){
+        await actions.wrappedFetchMetrics(this.props);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<BaseSidebar/>}
+            >
+                <MetricList {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/metrics/MetricQuestions.jsx b/frontend/src/metabase/reference/metrics/MetricQuestions.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..759abf19695807d4ab15594a7a55f754538b5ed8
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricQuestions.jsx
@@ -0,0 +1,118 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import moment from "moment";
+
+import visualizations from "metabase/visualizations";
+import { isQueryable } from "metabase/lib/table";
+import * as Urls from "metabase/lib/urls";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+import {
+    getMetricQuestions,
+    getError,
+    getLoading,
+    getTable,
+    getMetric
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+const emptyStateData = (table, metric) => {
+    return {
+        message: "Questions about this metric will appear here as they're added",
+        icon: "all",
+        action: "Ask a question",
+        link: getQuestionUrl({
+            dbId: table && table.db_id,
+            tableId: metric.table_id,
+            metricId: metric.id
+        })
+    };
+    }
+
+
+const mapStateToProps = (state, props) => ({
+    metric: getMetric(state, props),
+    table: getTable(state, props),
+    entities: getMetricQuestions(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricQuestions extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        metric: PropTypes.object,
+        table: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name={`Questions about ${this.props.metric.name}`}
+                    type="questions"
+                    headerIcon="ruler"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            { 
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                        <li className="relative" key={entity.id}>
+                                            <ListItem
+                                                id={entity.id}
+                                                index={index}
+                                                name={entity.display_name || entity.name}
+                                                description={ `Created ${moment(entity.created_at).fromNow()} by ${entity.creator.common_name}` }
+                                                url={ Urls.question(entity.id) }
+                                                icon={ visualizations.get(entity.display).iconName }
+                                            />
+                                        </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData(this.props.table, this.props.metric)}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/metrics/MetricQuestionsContainer.jsx b/frontend/src/metabase/reference/metrics/MetricQuestionsContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..6c00b2b626ef5d7619731d7a9983cc81c16ca1e4
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricQuestionsContainer.jsx
@@ -0,0 +1,84 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import MetricSidebar from './MetricSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import MetricQuestions from "metabase/reference/metrics/MetricQuestions.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getMetric,
+    getMetricId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+import {
+    loadEntities
+} from 'metabase/questions/questions';
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    metric: getMetric(state, props),
+    metricId: getMetricId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    fetchQuestions: () => loadEntities("cards", {}),
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricQuestionsContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        user: PropTypes.object.isRequired,
+        metric: PropTypes.object.isRequired,
+        metricId: PropTypes.number.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchMetricQuestions(this.props, this.props.metricId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            metric,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<MetricSidebar metric={metric} user={user}/>}
+            >
+                <MetricQuestions {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/containers/ReferenceRevisionsList.jsx b/frontend/src/metabase/reference/metrics/MetricRevisions.jsx
similarity index 87%
rename from frontend/src/metabase/reference/containers/ReferenceRevisionsList.jsx
rename to frontend/src/metabase/reference/metrics/MetricRevisions.jsx
index 05465289ba7289c2adf13f311541b27a6c790217..f9dcbf7af14725e4b5d3bcaabac14b3485720fce 100644
--- a/frontend/src/metabase/reference/containers/ReferenceRevisionsList.jsx
+++ b/frontend/src/metabase/reference/metrics/MetricRevisions.jsx
@@ -11,8 +11,7 @@ import * as metadataActions from "metabase/redux/metadata";
 import { assignUserColors } from "metabase/lib/formatting";
 
 import {
-    getSection,
-    getData,
+    getMetricRevisions,
     getMetric,
     getSegment,
     getTables,
@@ -26,10 +25,14 @@ import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.j
 import EmptyState from "metabase/components/EmptyState.jsx";
 import ReferenceHeader from "../components/ReferenceHeader.jsx";
 
+
+const emptyStateData =  {
+    message: "There are no revisions for this metric"
+}
+
 const mapStateToProps = (state, props) => {
     return {
-        section: getSection(state, props),
-        revisions: getData(state, props),
+        revisions: getMetricRevisions(state, props),
         metric: getMetric(state, props),
         segment: getSegment(state, props),
         tables: getTables(state, props),
@@ -44,10 +47,9 @@ const mapDispatchToProps = {
 };
 
 @connect(mapStateToProps, mapDispatchToProps)
-export default class RevisionHistoryApp extends Component {
+export default class MetricRevisions extends Component {
     static propTypes = {
         style: PropTypes.object.isRequired,
-        section: PropTypes.object.isRequired,
         revisions: PropTypes.object.isRequired,
         metric: PropTypes.object.isRequired,
         segment: PropTypes.object.isRequired,
@@ -60,7 +62,6 @@ export default class RevisionHistoryApp extends Component {
     render() {
         const {
             style,
-            section,
             revisions,
             metric,
             segment,
@@ -72,11 +73,6 @@ export default class RevisionHistoryApp extends Component {
 
         const entity = metric.id ? metric : segment;
 
-        const empty = {
-            icon: 'mine',
-            message: 'You haven\'t added any databases yet.'
-        };
-
         const userColorAssignments = user && Object.keys(revisions).length > 0 ?
             assignUserColors(
                 Object.values(revisions)
@@ -86,7 +82,10 @@ export default class RevisionHistoryApp extends Component {
 
         return (
             <div style={style} className="full">
-                <ReferenceHeader section={section} />
+                <ReferenceHeader 
+                    name={`Revision history for ${this.props.metric.name}`}
+                    headerIcon="ruler"
+                />
                 <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
                     { () => Object.keys(revisions).length > 0 && tables[entity.table_id] ?
                         <div className="wrapper wrapper--trim">
@@ -109,7 +108,7 @@ export default class RevisionHistoryApp extends Component {
                         </div>
                         :
                         <div className={S.empty}>
-                          <EmptyState message={empty.message} icon={empty.icon} />
+                          <EmptyState {...emptyStateData}/>
                         </div>
                     }
                 </LoadingAndErrorWrapper>
diff --git a/frontend/src/metabase/reference/metrics/MetricRevisionsContainer.jsx b/frontend/src/metabase/reference/metrics/MetricRevisionsContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..2c858ae51c2756450f9acda0395e72d7d4e7dfc0
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricRevisionsContainer.jsx
@@ -0,0 +1,82 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import MetricSidebar from './MetricSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import MetricRevisions from "metabase/reference/metrics/MetricRevisions.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getMetric,
+    getMetricId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    metric: getMetric(state, props),
+    metricId: getMetricId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class MetricRevisionsContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        user: PropTypes.object.isRequired,
+        metric: PropTypes.object.isRequired,
+        metricId: PropTypes.number.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+
+    async fetchContainerData(){
+        await actions.wrappedFetchMetricRevisions(this.props, this.props.metricId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            metric,
+            isEditing
+        } = this.props;
+
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<MetricSidebar metric={metric} user={user}/>}
+            >
+                <MetricRevisions {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/metrics/MetricSidebar.jsx b/frontend/src/metabase/reference/metrics/MetricSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..3803507470966b3e83b3aa63cd84480f4f467be6
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/MetricSidebar.jsx
@@ -0,0 +1,55 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const MetricSidebar = ({
+    metric,
+    user,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <ul>
+            <div className={S.breadcrumbs}>
+                <Breadcrumbs
+                    className="py4"
+                    crumbs={[["Metrics","/reference/metrics"],
+                             [metric.name]]}
+                    inSidebar={true}
+                    placeholder="Data Reference"
+                />
+            </div>
+                <SidebarItem key={`/reference/metrics/${metric.id}`} 
+                             href={`/reference/metrics/${metric.id}`} 
+                             icon="document" 
+                             name="Details" />
+                <SidebarItem key={`/reference/metrics/${metric.id}/questions`} 
+                             href={`/reference/metrics/${metric.id}/questions`} 
+                             icon="all" 
+                             name={`Questions about ${metric.name}`} />
+             { user && user.is_superuser &&
+
+                <SidebarItem key={`/reference/metrics/${metric.id}/revisions`}
+                             href={`/reference/metrics/${metric.id}/revisions`}
+                             icon="history" 
+                             name={`Revision history for ${metric.name}`} />
+             }
+        </ul>
+    </div>
+
+MetricSidebar.propTypes = {
+    metric:          PropTypes.object,
+    user:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(MetricSidebar);
+
diff --git a/frontend/src/metabase/reference/metrics/metrics.integ.spec.js b/frontend/src/metabase/reference/metrics/metrics.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..65f5d2da158fd32f9e532bcf0edd37c97d746f06
--- /dev/null
+++ b/frontend/src/metabase/reference/metrics/metrics.integ.spec.js
@@ -0,0 +1,127 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { CardApi, MetricApi } from 'metabase/services'
+
+import { 
+    FETCH_METRICS,
+    FETCH_METRIC_TABLE,
+    FETCH_METRIC_REVISIONS
+} from "metabase/redux/metadata";
+
+import { FETCH_GUIDE } from "metabase/reference/reference"
+
+import MetricListContainer from "metabase/reference/metrics/MetricListContainer";
+import MetricDetailContainer from "metabase/reference/metrics/MetricDetailContainer";
+import MetricQuestionsContainer from "metabase/reference/metrics/MetricQuestionsContainer";
+import MetricRevisionsContainer from "metabase/reference/metrics/MetricRevisionsContainer";
+
+
+describe("The Reference Section", () => {
+    // Test data
+    const metricDef = {name: "A Metric", description: "I did it!", table_id: 1,show_in_getting_started: true,
+                        definition: {database: 1, query: {aggregation: ["count"]}}}
+
+    const anotherMetricDef = {name: "Another Metric", description: "I did it again!", table_id: 1,show_in_getting_started: true,
+                        definition: {database: 1, query: {aggregation: ["count"]}}}
+
+    const metricCardDef = { name :"A card", display: "scalar", 
+                      dataset_query: {database: 1, table_id: 1, type: "query", query: {source_table: 1, "aggregation": ["metric", 1]}},
+                      visualization_settings: {}}
+
+    // Scaffolding
+    beforeAll(async () => {
+        await login();
+
+    })
+
+
+    
+    describe("The Metrics section of the Data Reference", async ()=>{
+        describe("Empty State", async () => {
+
+            it("Should show no metrics in the list", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/metrics");
+                mount(store.connectContainer(<MetricListContainer />));
+                await store.waitForActions([FETCH_METRICS])
+            })
+
+        });
+
+        describe("With Metrics State", async () => {
+            var metricIds = []
+
+            beforeAll(async () => {            
+                // Create some metrics to have something to look at
+                var metric = await MetricApi.create(metricDef);
+                var metric2 = await MetricApi.create(anotherMetricDef);
+                
+                metricIds.push(metric.id)
+                metricIds.push(metric2.id)
+                console.log(metricIds)
+                })
+
+            afterAll(async () => {
+                // Delete the guide we created
+                // remove the metrics we created   
+                // This is a bit messy as technically these are just archived
+                for (const id of metricIds){
+                    await MetricApi.delete({metricId: id, revision_message: "Please"})
+                }
+            })
+            // metrics list
+            it("Should show no metrics in the list", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/metrics");
+                mount(store.connectContainer(<MetricListContainer />));
+                await store.waitForActions([FETCH_METRICS])
+            })
+            // metric detail
+            it("Should show the metric detail view for a specific id", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/metrics/"+metricIds[0]);
+                mount(store.connectContainer(<MetricDetailContainer />));
+                await store.waitForActions([FETCH_METRIC_TABLE, FETCH_GUIDE])
+            })
+            // metrics questions 
+            it("Should show no questions based on a new metric", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/metrics/"+metricIds[0]+'/questions');
+                mount(store.connectContainer(<MetricQuestionsContainer />));
+                await store.waitForActions([FETCH_METRICS, FETCH_METRIC_TABLE])
+            })
+            // metrics revisions
+            it("Should show revisions", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/metrics/"+metricIds[0]+'/revisions');
+                mount(store.connectContainer(<MetricRevisionsContainer />));
+                await store.waitForActions([FETCH_METRICS, FETCH_METRIC_REVISIONS])
+            })
+
+            it("Should see a newly asked question in its questions list", async () => {
+                    var card = await CardApi.create(metricCardDef)
+
+                    expect(card.name).toBe(metricCardDef.name);
+                    // see that there is a new question on the metric's questions page
+                    const store = await createTestStore()    
+                    store.pushPath("/reference/metrics/"+metricIds[0]+'/questions');
+                    mount(store.connectContainer(<MetricQuestionsContainer />));
+                    await store.waitForActions([FETCH_METRICS, FETCH_METRIC_TABLE])
+                    
+                    await CardApi.delete({cardId: card.id})
+            })
+
+                       
+        });
+    });
+    
+
+
+
+});
\ No newline at end of file
diff --git a/frontend/src/metabase/reference/reference.js b/frontend/src/metabase/reference/reference.js
index 13269160cf62a1ab736d77fad7a4f4aa65f5590a..8c44378cb7ba8692a0b77ab746c3a9d451ed5cf3 100644
--- a/frontend/src/metabase/reference/reference.js
+++ b/frontend/src/metabase/reference/reference.js
@@ -1,4 +1,5 @@
 import { assoc } from 'icepick';
+import _ from "underscore";
 
 import {
     handleActions,
@@ -11,7 +12,24 @@ import MetabaseAnalytics from 'metabase/lib/analytics';
 
 import { GettingStartedApi } from "metabase/services";
 
-const FETCH_GUIDE = "metabase/reference/FETCH_GUIDE";
+import { 
+    filterUntouchedFields, 
+    isEmptyObject 
+} from "./utils.js"
+
+export const FETCH_GUIDE = "metabase/reference/FETCH_GUIDE";
+export const SET_ERROR = "metabase/reference/SET_ERROR";
+export const CLEAR_ERROR = "metabase/reference/CLEAR_ERROR";
+export const START_LOADING = "metabase/reference/START_LOADING";
+export const END_LOADING = "metabase/reference/END_LOADING";
+export const START_EDITING = "metabase/reference/START_EDITING";
+export const END_EDITING = "metabase/reference/END_EDITING";
+export const EXPAND_FORMULA = "metabase/reference/EXPAND_FORMULA";
+export const COLLAPSE_FORMULA = "metabase/reference/COLLAPSE_FORMULA";
+export const SHOW_DASHBOARD_MODAL = "metabase/reference/SHOW_DASHBOARD_MODAL";
+export const HIDE_DASHBOARD_MODAL = "metabase/reference/HIDE_DASHBOARD_MODAL";
+
+
 export const fetchGuide = createThunkAction(FETCH_GUIDE, (reload = false) => {
     return async (dispatch, getState) => {
         const requestStatePath = ["reference", 'guide'];
@@ -31,42 +49,499 @@ export const fetchGuide = createThunkAction(FETCH_GUIDE, (reload = false) => {
     };
 });
 
-const SET_ERROR = "metabase/reference/SET_ERROR";
 export const setError = createAction(SET_ERROR);
 
-const CLEAR_ERROR = "metabase/reference/CLEAR_ERROR";
 export const clearError = createAction(CLEAR_ERROR);
 
-const START_LOADING = "metabase/reference/START_LOADING";
 export const startLoading = createAction(START_LOADING);
 
-const END_LOADING = "metabase/reference/END_LOADING";
 export const endLoading = createAction(END_LOADING);
 
-const START_EDITING = "metabase/reference/START_EDITING";
 export const startEditing = createAction(START_EDITING, () => {
     MetabaseAnalytics.trackEvent('Data Reference', 'Started Editing');
 });
 
-const END_EDITING = "metabase/reference/END_EDITING";
 export const endEditing = createAction(END_EDITING, () => {
     MetabaseAnalytics.trackEvent('Data Reference', 'Ended Editing');
 });
 
-const EXPAND_FORMULA = "metabase/reference/EXPAND_FORMULA";
 export const expandFormula = createAction(EXPAND_FORMULA);
 
-const COLLAPSE_FORMULA = "metabase/reference/COLLAPSE_FORMULA";
 export const collapseFormula = createAction(COLLAPSE_FORMULA);
 
 //TODO: consider making an app-wide modal state reducer and related actions
-const SHOW_DASHBOARD_MODAL = "metabase/reference/SHOW_DASHBOARD_MODAL";
 export const showDashboardModal = createAction(SHOW_DASHBOARD_MODAL);
 
-const HIDE_DASHBOARD_MODAL = "metabase/reference/HIDE_DASHBOARD_MODAL";
 export const hideDashboardModal = createAction(HIDE_DASHBOARD_MODAL);
 
 
+// Helper functions. This is meant to be a transitional state to get things out of tryFetchData() and friends
+
+const fetchDataWrapper = (props, fn) => {
+
+    return async (argument) => {
+        props.clearError();
+        props.startLoading();
+        try {
+            await fn(argument)
+        }
+        catch(error) {
+            console.error(error);
+            props.setError(error);
+        }
+
+        props.endLoading();
+    }
+}
+export const wrappedFetchGuide = async (props) => {
+
+    fetchDataWrapper(
+        props, 
+        async () => { 
+                await Promise.all(
+                    [props.fetchGuide(),
+                     props.fetchDashboards(),
+                     props.fetchMetrics(),
+                     props.fetchSegments(),
+                     props.fetchDatabasesWithMetadata()]
+                )}
+        )()
+}
+export const wrappedFetchDatabaseMetadata = (props, databaseID) => {
+    fetchDataWrapper(props, props.fetchDatabaseMetadata)(databaseID)
+}
+
+export const wrappedFetchDatabaseMetadataAndQuestion = async (props, databaseID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (dbID) => { 
+                await Promise.all(
+                    [props.fetchDatabaseMetadata(dbID),
+                     props.fetchQuestions()]
+                )}
+        )(databaseID)
+}
+export const wrappedFetchMetricDetail = async (props, metricID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (mID) => { 
+                await Promise.all(
+                    [props.fetchMetricTable(mID),
+                     props.fetchMetrics(), 
+                     props.fetchGuide()]
+                )}
+        )(metricID)
+}
+export const wrappedFetchMetricQuestions = async (props, metricID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (mID) => { 
+                await Promise.all(
+                    [props.fetchMetricTable(mID),
+                     props.fetchMetrics(), 
+                     props.fetchQuestions()]
+                )}
+        )(metricID)
+}
+export const wrappedFetchMetricRevisions = async (props, metricID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (mID) => { 
+                await Promise.all(
+                    [props.fetchMetricRevisions(mID),
+                     props.fetchMetrics()]
+                )}
+        )(metricID)
+}
+
+// export const wrappedFetchDatabaseMetadataAndQuestion = async (props, databaseID) => {
+//         clearError();
+//         startLoading();
+//         try {
+//             await Promise.all(
+//                     [props.fetchDatabaseMetadata(databaseID),
+//                      props.fetchQuestions()]
+//                 )
+//         }
+//         catch(error) {
+//             console.error(error);
+//             setError(error);
+//         }
+
+//         endLoading();
+// }
+
+export const wrappedFetchDatabases = (props) => {
+    fetchDataWrapper(props, props.fetchDatabases)({})
+}
+export const wrappedFetchMetrics = (props) => {
+    fetchDataWrapper(props, props.fetchMetrics)({})
+}
+
+export const wrappedFetchSegments = (props) => {
+    fetchDataWrapper(props, props.fetchSegments)({})
+}
+
+
+export const wrappedFetchSegmentDetail = (props, segmentID) => {
+    fetchDataWrapper(props, props.fetchSegmentTable)(segmentID)
+}
+
+export const wrappedFetchSegmentQuestions = async (props, segmentID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (sID) => { 
+                await props.fetchSegments(sID);
+                await Promise.all(
+                    [props.fetchSegmentTable(sID),
+                     props.fetchQuestions()]
+                )}
+        )(segmentID)
+}
+export const wrappedFetchSegmentRevisions = async (props, segmentID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (sID) => { 
+                await props.fetchSegments(sID);
+                await Promise.all(
+                    [props.fetchSegmentRevisions(sID),
+                     props.fetchSegmentTable(sID)]
+                )}
+        )(segmentID)
+}
+export const wrappedFetchSegmentFields = async (props, segmentID) => {
+
+    fetchDataWrapper(
+        props, 
+        async (sID) => { 
+                await props.fetchSegments(sID);
+                await Promise.all(
+                    [props.fetchSegmentFields(sID),
+                     props.fetchSegmentTable(sID)]
+                )}
+        )(segmentID)
+}
+
+// This is called when a component gets a new set of props.
+// I *think* this is un-necessary in all cases as we're using multiple 
+// components where the old code re-used the same component
+export const clearState = props => {
+    props.endEditing();
+    props.endLoading();
+    props.clearError();
+    props.collapseFormula();
+}
+
+
+// This is called on the success or failure of a form triggered update
+const resetForm = (props) => {
+    props.resetForm();
+    props.endLoading();
+    props.endEditing();
+}
+
+// Update actions
+// these use the "fetchDataWrapper" for now. It should probably be renamed. 
+// Using props to fire off actions, which imo should be refactored to 
+// dispatch directly, since there is no actual dependence with the props 
+// of that component
+
+const updateDataWrapper = (props, fn) => {
+
+    return async (fields) => {
+        props.clearError();
+        props.startLoading();
+        try {
+            const editedFields = filterUntouchedFields(fields, props.entity);
+            if (!isEmptyObject(editedFields)) {
+                const newEntity = {...props.entity, ...editedFields};
+                await fn(newEntity);
+            }
+        }
+        catch(error) {
+            console.error(error);
+            props.setError(error);
+        }
+        resetForm(props)
+    }
+}
+
+export const rUpdateSegmentDetail = (formFields, props) => {
+    updateDataWrapper(props, props.updateSegment)(formFields)
+}
+export const rUpdateSegmentFieldDetail = (formFields, props) => {
+    updateDataWrapper(props, props.updateField)(formFields)
+}
+export const rUpdateDatabaseDetail = (formFields, props) => {
+    updateDataWrapper(props, props.updateDatabase)(formFields)
+}
+export const rUpdateTableDetail = (formFields, props) => {
+    updateDataWrapper(props, props.updateTable)(formFields)
+}
+export const rUpdateFieldDetail = (formFields, props) => {
+    updateDataWrapper(props, props.updateField)(formFields)
+}
+
+export const rUpdateMetricDetail = async (metric, guide, formFields, props) => {
+    props.startLoading();
+    try {
+        const editedFields = filterUntouchedFields(formFields, metric);
+        if (!isEmptyObject(editedFields)) {
+            const newMetric = {...metric, ...editedFields};
+            await props.updateMetric(newMetric);
+
+            const importantFieldIds = formFields.important_fields.map(field => field.id);
+            const existingImportantFieldIds = guide.metric_important_fields && guide.metric_important_fields[metric.id];
+
+            const areFieldIdsIdentitical = existingImportantFieldIds &&
+                existingImportantFieldIds.length === importantFieldIds.length &&
+                existingImportantFieldIds.every(id => importantFieldIds.includes(id));
+
+            if (!areFieldIdsIdentitical) {
+                await props.updateMetricImportantFields(metric.id, importantFieldIds);
+                wrappedFetchMetricDetail(props, metric.id);
+            }
+        }
+    }
+    catch(error) {
+        props.setError(error);
+        console.error(error);
+    }
+
+    resetForm(props)
+}
+
+export const rUpdateFields = async (fields, formFields, props) => {
+    props.startLoading();
+    try {
+        const updatedFields = Object.keys(formFields)
+            .map(fieldId => ({
+                field: fields[fieldId],
+                formField: filterUntouchedFields(formFields[fieldId], fields[fieldId])
+            }))
+            .filter(({field, formField}) => !isEmptyObject(formField))
+            .map(({field, formField}) => ({...field, ...formField}));
+
+        await Promise.all(updatedFields.map(props.updateField));
+    }
+    catch(error) {
+        props.setError(error);
+        console.error(error);
+    }
+
+    resetForm(props)
+}
+
+
+export const tryUpdateGuide = async (formFields, props) => {
+    const {
+        guide,
+        dashboards,
+        metrics,
+        segments,
+        tables,
+        startLoading,
+        endLoading,
+        endEditing,
+        setError,
+        resetForm,
+        updateDashboard,
+        updateMetric,
+        updateSegment,
+        updateTable,
+        updateMetricImportantFields,
+        updateSetting,
+        fetchGuide,
+        clearRequestState
+    } = props;
+
+    startLoading();
+    try {
+        const updateNewEntities = ({
+            entities,
+            formFields,
+            updateEntity
+        }) => formFields.map(formField => {
+            if (!formField.id) {
+                return [];
+            }
+
+            const editedEntity = filterUntouchedFields(
+                assoc(formField, 'show_in_getting_started', true),
+                entities[formField.id]
+            );
+
+            if (isEmptyObject(editedEntity)) {
+                return [];
+            }
+
+            const newEntity = entities[formField.id];
+            const updatedNewEntity = {
+                ...newEntity,
+                ...editedEntity
+            };
+
+            const updatingNewEntity = updateEntity(updatedNewEntity);
+
+            return [updatingNewEntity];
+        });
+
+        const updateOldEntities = ({
+            newEntityIds,
+            oldEntityIds,
+            entities,
+            updateEntity
+        }) => oldEntityIds
+            .filter(oldEntityId => !newEntityIds.includes(oldEntityId))
+            .map(oldEntityId => {
+                const oldEntity = entities[oldEntityId];
+
+                const updatedOldEntity = assoc(
+                    oldEntity,
+                    'show_in_getting_started',
+                    false
+                );
+
+                const updatingOldEntity = updateEntity(updatedOldEntity);
+
+                return [updatingOldEntity];
+            });
+        //FIXME: necessary because revision_message is a mandatory field
+        // even though we don't actually keep track of changes to caveats/points_of_interest yet
+        const updateWithRevisionMessage = updateEntity => entity => updateEntity(assoc(
+            entity,
+            'revision_message',
+            'Updated in Getting Started guide.'
+        ));
+
+        const updatingDashboards = updateNewEntities({
+                entities: dashboards,
+                formFields: [formFields.most_important_dashboard],
+                updateEntity: updateDashboard
+            })
+            .concat(updateOldEntities({
+                newEntityIds: formFields.most_important_dashboard ?
+                    [formFields.most_important_dashboard.id] : [],
+                oldEntityIds: guide.most_important_dashboard ?
+                    [guide.most_important_dashboard] :
+                    [],
+                entities: dashboards,
+                updateEntity: updateDashboard
+            }));
+
+        const updatingMetrics = updateNewEntities({
+                entities: metrics,
+                formFields: formFields.important_metrics,
+                updateEntity: updateWithRevisionMessage(updateMetric)
+            })
+            .concat(updateOldEntities({
+                newEntityIds: formFields.important_metrics
+                    .map(formField => formField.id),
+                oldEntityIds: guide.important_metrics,
+                entities: metrics,
+                updateEntity: updateWithRevisionMessage(updateMetric)
+            }));
+
+        const updatingMetricImportantFields = formFields.important_metrics
+            .map(metricFormField => {
+                if (!metricFormField.id || !metricFormField.important_fields) {
+                    return [];
+                }
+                const importantFieldIds = metricFormField.important_fields
+                    .map(field => field.id);
+                const existingImportantFieldIds = guide.metric_important_fields[metricFormField.id];
+
+                const areFieldIdsIdentitical = existingImportantFieldIds &&
+                    existingImportantFieldIds.length === importantFieldIds.length &&
+                    existingImportantFieldIds.every(id => importantFieldIds.includes(id));
+                if (areFieldIdsIdentitical) {
+                    return [];
+                }
+
+                return [updateMetricImportantFields(metricFormField.id, importantFieldIds)];
+            });
+
+        const segmentFields = formFields.important_segments_and_tables
+            .filter(field => field.type === 'segment');
+
+        const updatingSegments = updateNewEntities({
+                entities: segments,
+                formFields: segmentFields,
+                updateEntity: updateWithRevisionMessage(updateSegment)
+            })
+            .concat(updateOldEntities({
+                newEntityIds: segmentFields
+                    .map(formField => formField.id),
+                oldEntityIds: guide.important_segments,
+                entities: segments,
+                updateEntity: updateWithRevisionMessage(updateSegment)
+            }));
+
+        const tableFields = formFields.important_segments_and_tables
+            .filter(field => field.type === 'table');
+
+        const updatingTables = updateNewEntities({
+                entities: tables,
+                formFields: tableFields,
+                updateEntity: updateTable
+            })
+            .concat(updateOldEntities({
+                newEntityIds: tableFields
+                    .map(formField => formField.id),
+                oldEntityIds: guide.important_tables,
+                entities: tables,
+                updateEntity: updateTable
+            }));
+
+        const updatingThingsToKnow = guide.things_to_know !== formFields.things_to_know ?
+            [updateSetting({key: 'getting-started-things-to-know', value: formFields.things_to_know })] :
+            [];
+
+        const updatingContactName = guide.contact && formFields.contact &&
+            guide.contact.name !== formFields.contact.name ?
+                [updateSetting({key: 'getting-started-contact-name', value: formFields.contact.name })] :
+                [];
+
+        const updatingContactEmail = guide.contact && formFields.contact &&
+            guide.contact.email !== formFields.contact.email ?
+                [updateSetting({key: 'getting-started-contact-email', value: formFields.contact.email })] :
+                [];
+
+        const updatingData = _.flatten([
+            updatingDashboards,
+            updatingMetrics,
+            updatingMetricImportantFields,
+            updatingSegments,
+            updatingTables,
+            updatingThingsToKnow,
+            updatingContactName,
+            updatingContactEmail
+        ]);
+
+        if (updatingData.length > 0) {
+            await Promise.all(updatingData);
+
+            clearRequestState({statePath: ['reference', 'guide']});
+
+            await fetchGuide();
+        }
+    }
+    catch(error) {
+        setError(error);
+        console.error(error);
+    }
+
+    resetForm();
+    endLoading();
+    endEditing();
+};
+
 const initialState = {
     error: null,
     isLoading: false,
diff --git a/frontend/src/metabase/reference/segments/SegmentDetail.jsx b/frontend/src/metabase/reference/segments/SegmentDetail.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..4bce042b45f24f8363818a644ec17e41d5a680e9
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentDetail.jsx
@@ -0,0 +1,237 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+
+import List from "metabase/components/List.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+import Detail from "metabase/reference/components/Detail.jsx";
+import UsefulQuestions from "metabase/reference/components/UsefulQuestions.jsx";
+import Formula from "metabase/reference/components/Formula.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+import {
+    getSegment,
+    getTable,
+    getFields,
+    getGuide,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getIsFormulaExpanded,
+} from "../selectors";
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+const interestingQuestions = (table, segment) => {
+    return [
+        {
+            text: `Number of ${segment.name}`,
+            icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
+            link: getQuestionUrl({
+                dbId: table && table.db_id,
+                tableId: table.id,
+                segmentId: segment.id,
+                getCount: true
+            })
+        },
+        {
+            text: `See all ${segment.name}`,
+            icon: "table2",
+            link: getQuestionUrl({
+                dbId: table && table.db_id,
+                tableId: table.id,
+                segmentId: segment.id
+            })
+        }
+    ]
+}
+
+const mapStateToProps = (state, props) => {
+    const entity = getSegment(state, props) || {};
+    const guide = getGuide(state, props);
+    const fields = getFields(state, props);
+
+    const initialValues = {
+        important_fields: guide && guide.metric_important_fields &&
+            guide.metric_important_fields[entity.id] &&
+            guide.metric_important_fields[entity.id]
+                .map(fieldId => fields[fieldId]) ||
+                []
+    };
+
+    return {
+        entity,
+        table: getTable(state, props),
+        metadataFields: fields,
+        guide,
+        loading: getLoading(state, props),
+        // naming this 'error' will conflict with redux form
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        isEditing: getIsEditing(state, props),
+        isFormulaExpanded: getIsFormulaExpanded(state, props),
+        initialValues,
+    }
+};
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions,
+};
+
+const validate = (values, props) =>  !values.revision_message ? 
+    { revision_message: "Please enter a revision message" } : {} 
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'details',
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats'],
+    validate
+})
+export default class SegmentDetail extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entity: PropTypes.object.isRequired,
+        table: PropTypes.object,
+        user: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        expandFormula: PropTypes.func.isRequired,
+        collapseFormula: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        resetForm: PropTypes.func.isRequired,
+        fields: PropTypes.object.isRequired,
+        isFormulaExpanded: PropTypes.bool,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool,
+    };
+
+    render() {
+        const {
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats },
+            style,
+            entity,
+            table,
+            loadingError,
+            loading,
+            user,
+            isEditing,
+            startEditing,
+            endEditing,
+            expandFormula,
+            collapseFormula,
+            isFormulaExpanded,
+            handleSubmit,
+            resetForm,
+            submitting,
+        } = this.props;
+
+        const onSubmit = handleSubmit(async (fields) =>
+            await actions.rUpdateSegmentDetail(fields, this.props)
+        );
+
+        return (
+            <form style={style} className="full"
+                onSubmit={onSubmit}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={true}
+                        onSubmit={onSubmit}
+                        endEditing={endEditing}
+                        reinitializeForm={resetForm}
+                        submitting={submitting}
+                        revisionMessageFormField={revision_message}
+                    />
+                }
+                <EditableReferenceHeader
+                    entity={entity}
+                    table={table}
+                    type="segment"
+                    headerIcon="segment"
+                    headerLink={getQuestionUrl({ dbId: table&&table.db_id, tableId: entity.table_id, segmentId: entity.id})}
+                    name="Details"
+                    user={user}
+                    isEditing={isEditing}
+                    hasSingleSchema={false}
+                    hasDisplayName={false}
+                    startEditing={startEditing}
+                    displayNameFormField={display_name}
+                    nameFormField={name}
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () =>
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            <li className="relative">
+                                <Detail
+                                    id="description"
+                                    name="Description"
+                                    description={entity.description}
+                                    placeholder="No description yet"
+                                    isEditing={isEditing}
+                                    field={description}
+                                />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="points_of_interest"
+                                    name={`Why this Segment is interesting`}
+                                    description={entity.points_of_interest}
+                                    placeholder="Nothing interesting yet"
+                                    isEditing={isEditing}
+                                    field={points_of_interest}
+                                    />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="caveats"
+                                    name={`Things to be aware of about this Segment`}
+                                    description={entity.caveats}
+                                    placeholder="Nothing to be aware of yet"
+                                    isEditing={isEditing}
+                                    field={caveats}
+                                />
+                            </li>
+                            { table && !isEditing &&
+                                <li className="relative">
+                                    <Formula
+                                        type="segment"
+                                        entity={entity}
+                                        table={table}
+                                        isExpanded={isFormulaExpanded}
+                                        expandFormula={expandFormula}
+                                        collapseFormula={collapseFormula}
+                                    />
+                                </li>
+                            }
+                            { !isEditing &&
+                                <li className="relative">
+                                    <UsefulQuestions questions={interestingQuestions(this.props.table, this.props.entity)} />
+                                </li>
+                            }
+                        </List>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentDetailContainer.jsx b/frontend/src/metabase/reference/segments/SegmentDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..1bb72f4b2d831aff28006a34c5760878ccc065c7
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentDetailContainer.jsx
@@ -0,0 +1,80 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import SegmentSidebar from './SegmentSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentDetail from "metabase/reference/segments/SegmentDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getSegment,
+    getSegmentId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    segment: getSegment(state, props),
+    segmentId: getSegmentId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        user: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        segmentId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegmentDetail(this.props, this.props.segmentId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            segment,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<SegmentSidebar segment={segment} user={user}/>}
+            >
+                <SegmentDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentFieldDetail.jsx b/frontend/src/metabase/reference/segments/SegmentFieldDetail.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..bfc652b22a2b1de0817640a35a3aa8ec3b693610
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentFieldDetail.jsx
@@ -0,0 +1,256 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+
+import S from "metabase/reference/Reference.css";
+
+import List from "metabase/components/List.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+import Detail from "metabase/reference/components/Detail.jsx";
+import FieldTypeDetail from "metabase/reference/components/FieldTypeDetail.jsx";
+import UsefulQuestions from "metabase/reference/components/UsefulQuestions.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+import {
+    getFieldBySegment,
+    getTable,
+    getFields,
+    getGuide,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getForeignKeys,
+    getIsFormulaExpanded
+} from "../selectors";
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+const interestingQuestions = (table, field) => {
+    return [
+        {
+            text: `Number of ${table && table.display_name} grouped by ${field.display_name}`,
+            icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
+            link: getQuestionUrl({
+                dbId: table && table.db_id,
+                tableId: table.id,
+                fieldId: field.id,
+                getCount: true
+            })
+        },
+        {
+            text: `All distinct values of ${field.display_name}`,
+            icon: "table2",
+            link: getQuestionUrl({
+                dbId: table && table.db_id,
+                tableId: table.id,
+                fieldId: field.id
+            })
+        }
+    ]
+}
+
+const mapStateToProps = (state, props) => {
+    const entity = getFieldBySegment(state, props) || {};
+    const guide = getGuide(state, props);
+    const fields = getFields(state, props);
+
+    const initialValues = {
+        important_fields: guide && guide.metric_important_fields &&
+            guide.metric_important_fields[entity.id] &&
+            guide.metric_important_fields[entity.id]
+                .map(fieldId => fields[fieldId]) ||
+                []
+    };
+
+    return {
+        entity,
+        table: getTable(state, props),
+        guide,
+        loading: getLoading(state, props),
+        // naming this 'error' will conflict with redux form
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        foreignKeys: getForeignKeys(state, props),
+        isEditing: getIsEditing(state, props),
+        isFormulaExpanded: getIsFormulaExpanded(state, props),
+        initialValues,
+    }
+};
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+
+const validate = (values, props) => {
+    return {};
+}
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'details',
+    fields: ['name', 'display_name', 'description', 'revision_message', 'points_of_interest', 'caveats',  'special_type', 'fk_target_field_id'],
+    validate
+})
+export default class SegmentFieldDetail extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entity: PropTypes.object.isRequired,
+        table: PropTypes.object,
+        user: PropTypes.object.isRequired,
+        foreignKeys: PropTypes.object,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        resetForm: PropTypes.func.isRequired,
+        fields: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool,
+    };
+
+    render() {
+        const {
+            fields: { name, display_name, description, revision_message, points_of_interest, caveats, special_type, fk_target_field_id },
+            style,
+            entity,
+            table,
+            loadingError,
+            loading,
+            user,
+            foreignKeys,
+            isEditing,
+            startEditing,
+            endEditing,
+            handleSubmit,
+            resetForm,
+            submitting,
+        } = this.props;
+
+        const onSubmit = handleSubmit(async (fields) =>
+            await actions.rUpdateSegmentFieldDetail(fields, this.props)
+        );
+
+        return (
+            <form style={style} className="full"
+                onSubmit={onSubmit}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={false}
+                        onSubmit={onSubmit}
+                        endEditing={endEditing}
+                        reinitializeForm={resetForm}
+                        submitting={submitting}
+                        revisionMessageFormField={revision_message}
+                    />
+                }
+                <EditableReferenceHeader
+                    entity={entity}
+                    table={table}
+                    headerIcon="field"
+                    name="Details"
+                    type="field"
+                    user={user}
+                    isEditing={isEditing}
+                    hasSingleSchema={false}
+                    hasDisplayName={true}
+                    startEditing={startEditing}
+                    displayNameFormField={display_name}
+                    nameFormField={name}
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () =>
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            <li className="relative">
+                                <Detail
+                                    id="description"
+                                    name="Description"
+                                    description={entity.description}
+                                    placeholder="No description yet"
+                                    isEditing={isEditing}
+                                    field={description}
+                                />
+                            </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <Detail
+                                        id="name"
+                                        name="Actual name in database"
+                                        description={entity.name}
+                                        subtitleClass={S.tableActualName}
+                                    />
+                                </li>
+                            }
+                            <li className="relative">
+                                <Detail
+                                    id="points_of_interest"
+                                    name={`Why this field is interesting`}
+                                    description={entity.points_of_interest}
+                                    placeholder="Nothing interesting yet"
+                                    isEditing={isEditing}
+                                    field={points_of_interest}
+                                    />
+                            </li>
+                            <li className="relative">
+                                <Detail
+                                    id="caveats"
+                                    name={`Things to be aware of about this field`}
+                                    description={entity.caveats}
+                                    placeholder="Nothing to be aware of yet"
+                                    isEditing={isEditing}
+                                    field={caveats}
+                                />
+                            </li>
+
+
+                            { !isEditing && 
+                                <li className="relative">
+                                    <Detail
+                                        id="base_type"
+                                        name={`Data type`}
+                                        description={entity.base_type}
+                                    />
+                                </li>
+                            }
+                                <li className="relative">
+                                    <FieldTypeDetail
+                                        field={entity}
+                                        foreignKeys={foreignKeys}
+                                        fieldTypeFormField={special_type}
+                                        foreignKeyFormField={fk_target_field_id}
+                                        isEditing={isEditing}
+                                    />
+                                </li>
+                            { !isEditing &&
+                                <li className="relative">
+                                    <UsefulQuestions questions={interestingQuestions(this.props.table, this.props.entity)} />
+                                </li>
+                            }
+
+
+                        </List>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentFieldDetailContainer.jsx b/frontend/src/metabase/reference/segments/SegmentFieldDetailContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..f0379db76a481da43144ad55d5521c5a2bb71aff
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentFieldDetailContainer.jsx
@@ -0,0 +1,80 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import SegmentFieldSidebar from './SegmentFieldSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentFieldDetail from "metabase/reference/segments/SegmentFieldDetail.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getSegment,
+    getSegmentId,
+    getField,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    segment: getSegment(state, props),    
+    segmentId: getSegmentId(state, props),
+    field: getField(state, props),    
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentFieldDetailContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        segment: PropTypes.object.isRequired,
+        segmentId: PropTypes.number.isRequired,
+        field: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegmentFields(this.props, this.props.segmentId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            segment,
+            field,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<SegmentFieldSidebar segment={segment} field={field}/>}
+            >
+                <SegmentFieldDetail {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentFieldList.jsx b/frontend/src/metabase/reference/segments/SegmentFieldList.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..b2c25f9308f521b6326d8cdb8478394032213d93
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentFieldList.jsx
@@ -0,0 +1,178 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import { reduxForm } from "redux-form";
+
+import S from "metabase/components/List.css";
+import R from "metabase/reference/Reference.css";
+import F from "metabase/reference/components/Field.css"
+
+import Field from "metabase/reference/components/Field.jsx";
+import List from "metabase/components/List.jsx";
+import EmptyState from "metabase/components/EmptyState.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import EditHeader from "metabase/reference/components/EditHeader.jsx";
+import EditableReferenceHeader from "metabase/reference/components/EditableReferenceHeader.jsx";
+
+import cx from "classnames";
+
+import {
+    getFieldsBySegment,
+    getForeignKeys,
+    getError,
+    getLoading,
+    getUser,
+    getIsEditing,
+    getSegment
+} from "../selectors";
+
+import {
+    fieldsToFormFields
+} from '../utils';
+
+import { getIconForField } from "metabase/lib/schema_metadata";
+
+import * as metadataActions from "metabase/redux/metadata";
+import * as actions from 'metabase/reference/reference';
+
+
+const emptyStateData = {
+    message: `Fields in this table will appear here as they're added`,
+    icon: "fields"
+}
+
+
+const mapStateToProps = (state, props) => {
+    const data = getFieldsBySegment(state, props);
+    return {
+        segment: getSegment(state,props),
+        entities: data,
+        foreignKeys: getForeignKeys(state, props),
+        loading: getLoading(state, props),
+        loadingError: getError(state, props),
+        user: getUser(state, props),
+        isEditing: getIsEditing(state, props),
+        fields: fieldsToFormFields(data)
+    };
+}
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+const validate = (values, props) => {
+    return {};
+}
+
+@connect(mapStateToProps, mapDispatchToProps)
+@reduxForm({
+    form: 'fields',
+    validate
+})
+export default class SegmentFieldList extends Component {
+    static propTypes = {
+        segment: PropTypes.object.isRequired,
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        foreignKeys: PropTypes.object.isRequired,
+        isEditing: PropTypes.bool,
+        startEditing: PropTypes.func.isRequired,
+        endEditing: PropTypes.func.isRequired,
+        startLoading: PropTypes.func.isRequired,
+        endLoading: PropTypes.func.isRequired,
+        setError: PropTypes.func.isRequired,
+        updateField: PropTypes.func.isRequired,
+        handleSubmit: PropTypes.func.isRequired,
+        user: PropTypes.object.isRequired,
+        fields: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object,
+        submitting: PropTypes.bool,
+        resetForm: PropTypes.func
+    };
+
+    render() {
+        const {
+            segment,
+            style,
+            entities,
+            fields,
+            foreignKeys,
+            loadingError,
+            loading,
+            user,
+            isEditing,
+            startEditing,
+            endEditing,
+            resetForm,
+            handleSubmit,
+            submitting
+        } = this.props;
+
+        return (
+            <form style={style} className="full"
+                onSubmit={handleSubmit(async (formFields) =>
+                    await actions.rUpdateFields(this.props.entities, formFields, this.props)
+                )}
+            >
+                { isEditing &&
+                    <EditHeader
+                        hasRevisionHistory={false}
+                        reinitializeForm={resetForm}
+                        endEditing={endEditing}
+                        submitting={submitting}
+                    />
+                }
+                <EditableReferenceHeader 
+                    type="segment"
+                    headerIcon="segment"
+                    name={`Fields in ${segment.name}`}
+                    user={user} 
+                    isEditing={isEditing} 
+                    startEditing={startEditing} 
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <div className={S.item}>
+                            <div className={R.columnHeader}>
+                                <div className={cx(S.itemTitle, F.fieldNameTitle)}>
+                                    Field name
+                                </div>
+                                <div className={cx(S.itemTitle, F.fieldType)}>
+                                    Field type
+                                </div>
+                                <div className={cx(S.itemTitle, F.fieldDataType)}>
+                                    Data type
+                                </div>
+                            </div>
+                        </div>
+                        <List>
+                            { Object.values(entities).map(entity =>
+                                entity && entity.id && entity.name &&
+                                    <li className="relative" key={entity.id}>
+                                        <Field
+                                            field={entity}
+                                            foreignKeys={foreignKeys}
+                                            url={`/reference/segments/${segment.id}/fields/${entity.id}`}
+                                            icon={getIconForField(entity)}
+                                            isEditing={isEditing}
+                                            formField={fields[entity.id]}
+                                        />
+                                    </li>
+                            )}
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <EmptyState {...emptyStateData}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </form>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentFieldListContainer.jsx b/frontend/src/metabase/reference/segments/SegmentFieldListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..5834d67c564e4f071dfc6553ed459c6cc77cfaa1
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentFieldListContainer.jsx
@@ -0,0 +1,79 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import SegmentSidebar from './SegmentSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentFieldList from "metabase/reference/segments/SegmentFieldList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getSegment,
+    getSegmentId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    segment: getSegment(state, props),
+    segmentId: getSegmentId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentFieldListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        user: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        segmentId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegmentFields(this.props, this.props.segmentId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            segment,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<SegmentSidebar segment={segment} user={user}/>}
+            >
+                <SegmentFieldList {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentFieldSidebar.jsx b/frontend/src/metabase/reference/segments/SegmentFieldSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..3a4788f09428752a5dfb9965a0464073c1f19684
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentFieldSidebar.jsx
@@ -0,0 +1,45 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const SegmentFieldSidebar = ({
+    segment,
+    field,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <ul>
+            <div className={S.breadcrumbs}>
+                <Breadcrumbs
+                    className="py4"
+                    crumbs={[["Segments","/reference/segments"],
+                             [segment.name, `/reference/segments/${segment.id}`],
+                             [field.name]]}
+                    inSidebar={true}
+                    placeholder="Data Reference"
+                />
+            </div>
+                <SidebarItem key={`/reference/segments/${segment.id}/fields/${field.id}`} 
+                             href={`/reference/segments/${segment.id}/fields/${field.id}`} 
+                             icon="document" 
+                             name="Details" />
+        </ul>
+    </div>
+
+SegmentFieldSidebar.propTypes = {
+    segment:          PropTypes.object,
+    field:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(SegmentFieldSidebar);
+
diff --git a/frontend/src/metabase/reference/segments/SegmentList.jsx b/frontend/src/metabase/reference/segments/SegmentList.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..66ff03ffa4b3af1e8150fbce4bbaff9088716666
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentList.jsx
@@ -0,0 +1,99 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+
+import { isQueryable } from "metabase/lib/table";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getSegments,
+    getError,
+    getLoading
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+const emptyStateData = {
+            title: "Segments are interesting subsets of tables",
+            adminMessage: "Defining common segments for your team makes it even easier to ask questions",
+            message: "Segments will appear here once your admins have created some",
+            image: "app/assets/img/segments-list",
+            adminAction: "Learn how to create segments",
+            adminLink: "http://www.metabase.com/docs/latest/administration-guide/06-segments-and-metrics.html"
+        }
+
+const mapStateToProps = (state, props) => ({
+    entities: getSegments(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentList extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name="Segments"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            {
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                         <li className="relative" key={entity.id}>
+                                            <ListItem
+                                                id={entity.id}
+                                                index={index}
+                                                name={entity.display_name || entity.name}
+                                                description={ entity.description }
+                                                url={ `/reference/segments/${entity.id}` }
+                                                icon="segment"
+                                            />
+                                        </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData}/>
+                        }
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentListContainer.jsx b/frontend/src/metabase/reference/segments/SegmentListContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..2516af6d97256bf95ef266ccc5093a9b7e1caf2a
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentListContainer.jsx
@@ -0,0 +1,70 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import BaseSidebar from 'metabase/reference/guide/BaseSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentList from "metabase/reference/segments/SegmentList.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentListContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegments(this.props);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<BaseSidebar/>}
+            >
+                <SegmentList {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentQuestions.jsx b/frontend/src/metabase/reference/segments/SegmentQuestions.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..ffef951c2565956d76af291d8121f4d247c6019b
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentQuestions.jsx
@@ -0,0 +1,116 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+import moment from "moment";
+
+import visualizations from "metabase/visualizations";
+import { isQueryable } from "metabase/lib/table";
+import * as Urls from "metabase/lib/urls";
+
+import S from "metabase/components/List.css";
+
+import List from "metabase/components/List.jsx";
+import ListItem from "metabase/components/ListItem.jsx";
+import AdminAwareEmptyState from "metabase/components/AdminAwareEmptyState.jsx";
+
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+import {
+    getQuestionUrl
+} from '../utils';
+
+
+import {
+    getSegmentQuestions,
+    getError,
+    getLoading,
+    getTableBySegment,
+    getSegment
+} from "../selectors";
+
+import * as metadataActions from "metabase/redux/metadata";
+
+const emptyStateData = (table, segment) =>{  
+    return {
+        message: "Questions about this segment will appear here as they're added",
+        icon: "all",
+        action: "Ask a question",
+        link: getQuestionUrl({
+            dbId: table && table.db_id,
+            tableId: segment.table_id,
+            segmentId: segment.id
+        })
+    };
+}
+const mapStateToProps = (state, props) => ({
+    segment: getSegment(state,props),
+    table: getTableBySegment(state,props),
+    entities: getSegmentQuestions(state, props),
+    loading: getLoading(state, props),
+    loadingError: getError(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentQuestions extends Component {
+    static propTypes = {
+        table: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        style: PropTypes.object.isRequired,
+        entities: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            entities,
+            style,
+            loadingError,
+            loading
+        } = this.props;
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name={`Questions about ${this.props.segment.name}`}
+                    type='questions'
+                    headerIcon="segment"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                { () => Object.keys(entities).length > 0 ?
+                    <div className="wrapper wrapper--trim">
+                        <List>
+                            { 
+                                Object.values(entities).filter(isQueryable).map((entity, index) =>
+                                    entity && entity.id && entity.name &&
+                                            <li className="relative" key={entity.id}>
+                                                <ListItem
+                                                    id={entity.id}
+                                                    index={index}
+                                                    name={entity.display_name || entity.name}
+                                                    description={ `Created ${moment(entity.created_at).fromNow()} by ${entity.creator.common_name}` }
+                                                    url={ Urls.question(entity.id) }
+                                                    icon={ visualizations.get(entity.display).iconName }
+                                                />
+                                            </li>
+                                )
+                            }
+                        </List>
+                    </div>
+                    :
+                    <div className={S.empty}>
+                        <AdminAwareEmptyState {...emptyStateData(this.props.table, this.props.segment)}/>
+                    </div>
+                }
+                </LoadingAndErrorWrapper>
+            </div>
+        )
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentQuestionsContainer.jsx b/frontend/src/metabase/reference/segments/SegmentQuestionsContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..db68e6558414fddcced0f552199be5cdb2824a18
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentQuestionsContainer.jsx
@@ -0,0 +1,85 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import SegmentSidebar from './SegmentSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentQuestions from "metabase/reference/segments/SegmentQuestions.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getSegment,
+    getSegmentId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+import {
+    loadEntities
+} from 'metabase/questions/questions';
+
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    segment: getSegment(state, props),
+    segmentId: getSegmentId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    fetchQuestions: () => loadEntities("cards", {}),
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentQuestionsContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        user: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        segmentId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegmentQuestions(this.props, this.props.segmentId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            segment,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<SegmentSidebar segment={segment} user={user}/>}
+            >
+                <SegmentQuestions {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentRevisions.jsx b/frontend/src/metabase/reference/segments/SegmentRevisions.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..b4a8d396e415d02f7a3438ac0a1f04b125f987c1
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentRevisions.jsx
@@ -0,0 +1,117 @@
+import React, { Component } from "react";
+import PropTypes from "prop-types";
+import { connect } from "react-redux";
+
+import { getIn } from "icepick";
+
+import S from "metabase/components/List.css";
+import R from "metabase/reference/Reference.css";
+
+import * as metadataActions from "metabase/redux/metadata";
+import { assignUserColors } from "metabase/lib/formatting";
+
+import {
+    getSegmentRevisions,
+    getMetric,
+    getSegment,
+    getTables,
+    getUser,
+    getLoading,
+    getError
+} from "../selectors";
+
+import Revision from "metabase/admin/datamodel/components/revisions/Revision.jsx";
+import LoadingAndErrorWrapper from "metabase/components/LoadingAndErrorWrapper.jsx";
+import EmptyState from "metabase/components/EmptyState.jsx";
+import ReferenceHeader from "../components/ReferenceHeader.jsx";
+
+const emptyStateData =  {
+    message: "There are no revisions for this segment"
+}
+
+const mapStateToProps = (state, props) => {
+    return {
+        revisions: getSegmentRevisions(state, props),
+        metric: getMetric(state, props),
+        segment: getSegment(state, props),
+        tables: getTables(state, props),
+        user: getUser(state, props),
+        loading: getLoading(state, props),
+        loadingError: getError(state, props)
+    }
+}
+
+const mapDispatchToProps = {
+    ...metadataActions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentRevisions extends Component {
+    static propTypes = {
+        style: PropTypes.object.isRequired,
+        revisions: PropTypes.object.isRequired,
+        metric: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        tables: PropTypes.object.isRequired,
+        user: PropTypes.object.isRequired,
+        loading: PropTypes.bool,
+        loadingError: PropTypes.object
+    };
+
+    render() {
+        const {
+            style,
+            revisions,
+            metric,
+            segment,
+            tables,
+            user,
+            loading,
+            loadingError
+        } = this.props;
+
+        const entity = metric.id ? metric : segment;
+
+        const userColorAssignments = user && Object.keys(revisions).length > 0 ?
+            assignUserColors(
+                Object.values(revisions)
+                    .map(revision => getIn(revision, ['user', 'id'])),
+                user.id
+            ) : {};
+
+        return (
+            <div style={style} className="full">
+                <ReferenceHeader 
+                    name={`Revision history for ${this.props.segment.name}`}
+                    headerIcon="segment"
+                />
+                <LoadingAndErrorWrapper loading={!loadingError && loading} error={loadingError}>
+                    { () => Object.keys(revisions).length > 0 && tables[entity.table_id] ?
+                        <div className="wrapper wrapper--trim">
+                            <div className={R.revisionsWrapper}>
+                                {Object.values(revisions)
+                                    .map(revision => revision && revision.diff ?
+                                        <Revision
+                                            key={revision.id}
+                                            revision={revision || {}}
+                                            tableMetadata={tables[entity.table_id] || {}}
+                                            objectName={entity.name}
+                                            currentUser={user || {}}
+                                            userColor={userColorAssignments[getIn(revision, ['user', 'id'])]}
+                                        /> :
+                                        null
+                                    )
+                                    .reverse()
+                                }
+                            </div>
+                        </div>
+                        :
+                        <div className={S.empty}>
+                          <EmptyState {...emptyStateData}/>
+                        </div>
+                    }
+                </LoadingAndErrorWrapper>
+            </div>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentRevisionsContainer.jsx b/frontend/src/metabase/reference/segments/SegmentRevisionsContainer.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..a74062fab583a13eef478d13356d2866f3d04045
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentRevisionsContainer.jsx
@@ -0,0 +1,81 @@
+/* eslint "react/prop-types": "warn" */
+import React, { Component } from 'react';
+import PropTypes from "prop-types";
+import { connect } from 'react-redux';
+
+import SegmentSidebar from './SegmentSidebar.jsx';
+import SidebarLayout from 'metabase/components/SidebarLayout.jsx';
+import SegmentRevisions from "metabase/reference/segments/SegmentRevisions.jsx"
+
+import * as metadataActions from 'metabase/redux/metadata';
+import * as actions from 'metabase/reference/reference';
+
+import {
+    getUser,
+    getSegment,
+    getSegmentId,
+    getDatabaseId,
+    getIsEditing
+} from '../selectors';
+
+
+const mapStateToProps = (state, props) => ({
+    user: getUser(state, props),
+    segment: getSegment(state, props),
+    segmentId: getSegmentId(state, props),
+    databaseId: getDatabaseId(state, props),
+    isEditing: getIsEditing(state, props)
+});
+
+const mapDispatchToProps = {
+    ...metadataActions,
+    ...actions
+};
+
+@connect(mapStateToProps, mapDispatchToProps)
+export default class SegmentRevisionsContainer extends Component {
+    static propTypes = {
+        params: PropTypes.object.isRequired,
+        location: PropTypes.object.isRequired,
+        databaseId: PropTypes.number.isRequired,
+        user: PropTypes.object.isRequired,
+        segment: PropTypes.object.isRequired,
+        segmentId: PropTypes.number.isRequired,
+        isEditing: PropTypes.bool
+    };
+
+    async fetchContainerData(){
+        await actions.wrappedFetchSegmentRevisions(this.props, this.props.segmentId);
+    }
+
+    componentWillMount() {
+        this.fetchContainerData()
+    }
+
+
+    componentWillReceiveProps(newProps) {
+        if (this.props.location.pathname === newProps.location.pathname) {
+            return;
+        }
+
+        actions.clearState(newProps)
+    }
+
+    render() {
+        const {
+            user,
+            segment,
+            isEditing
+        } = this.props;
+
+        return (
+            <SidebarLayout
+                className="flex-full relative"
+                style={ isEditing ? { paddingTop: '43px' } : {}}
+                sidebar={<SegmentSidebar segment={segment} user={user}/>}
+            >
+                <SegmentRevisions {...this.props} />
+            </SidebarLayout>
+        );
+    }
+}
diff --git a/frontend/src/metabase/reference/segments/SegmentSidebar.jsx b/frontend/src/metabase/reference/segments/SegmentSidebar.jsx
new file mode 100644
index 0000000000000000000000000000000000000000..fc955e1a6a0045fd57ffcd8af3a97a5037556d9b
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/SegmentSidebar.jsx
@@ -0,0 +1,59 @@
+/* eslint "react/prop-types": "warn" */
+import React from "react";
+import PropTypes from "prop-types";
+import S from "metabase/components/Sidebar.css";
+
+import Breadcrumbs from "metabase/components/Breadcrumbs.jsx";
+import SidebarItem from "metabase/components/SidebarItem.jsx"
+
+import cx from 'classnames';
+import pure from "recompose/pure";
+
+const SegmentSidebar = ({
+    segment,
+    user,
+    style,
+    className
+}) =>
+    <div className={cx(S.sidebar, className)} style={style}>
+        <ul>
+            <div className={S.breadcrumbs}>
+                <Breadcrumbs
+                    className="py4"
+                    crumbs={[["Segments","/reference/segments"],
+                             [segment.name]]}
+                    inSidebar={true}
+                    placeholder="Data Reference"
+                />
+            </div>
+                <SidebarItem key={`/reference/segments/${segment.id}`} 
+                             href={`/reference/segments/${segment.id}`} 
+                             icon="document" 
+                             name="Details" />
+                <SidebarItem key={`/reference/segments/${segment.id}/fields`} 
+                             href={`/reference/segments/${segment.id}/fields`} 
+                             icon="fields" 
+                             name="Fields in this segment" />
+                <SidebarItem key={`/reference/segments/${segment.id}/questions`} 
+                             href={`/reference/segments/${segment.id}/questions`} 
+                             icon="all" 
+                             name={`Questions about this segment`} />
+             { user && user.is_superuser &&
+
+                <SidebarItem key={`/reference/segments/${segment.id}/revisions`}
+                             href={`/reference/segments/${segment.id}/revisions`}
+                             icon="history" 
+                             name={`Revision history`} />
+             }
+        </ul>
+    </div>
+
+SegmentSidebar.propTypes = {
+    segment:          PropTypes.object,
+    user:          PropTypes.object,
+    className:      PropTypes.string,
+    style:          PropTypes.object,
+};
+
+export default pure(SegmentSidebar);
+
diff --git a/frontend/src/metabase/reference/segments/segments.integ.spec.js b/frontend/src/metabase/reference/segments/segments.integ.spec.js
new file mode 100644
index 0000000000000000000000000000000000000000..ba3435adb0df42b6f6c86bcd5f634ccefc5d4a3f
--- /dev/null
+++ b/frontend/src/metabase/reference/segments/segments.integ.spec.js
@@ -0,0 +1,148 @@
+import {
+    login,
+    createTestStore
+} from "metabase/__support__/integrated_tests";
+
+import React from 'react';
+import { mount } from 'enzyme';
+
+import { CardApi, SegmentApi } from 'metabase/services'
+
+import { 
+    FETCH_SEGMENTS,
+    FETCH_SEGMENT_TABLE,
+    FETCH_SEGMENT_FIELDS,
+    FETCH_SEGMENT_REVISIONS
+} from "metabase/redux/metadata";
+
+import { LOAD_ENTITIES } from "metabase/questions/questions"
+
+
+import SegmentListContainer from "metabase/reference/segments/SegmentListContainer";
+import SegmentDetailContainer from "metabase/reference/segments/SegmentDetailContainer";
+import SegmentQuestionsContainer from "metabase/reference/segments/SegmentQuestionsContainer";
+import SegmentRevisionsContainer from "metabase/reference/segments/SegmentRevisionsContainer";
+import SegmentFieldListContainer from "metabase/reference/segments/SegmentFieldListContainer";
+import SegmentFieldDetailContainer from "metabase/reference/segments/SegmentFieldDetailContainer";
+
+describe("The Reference Section", () => {
+    // Test data
+    const segmentDef = {name: "A Segment", description: "I did it!", table_id: 1, show_in_getting_started: true,
+                        definition: {database: 1, query: {filter: ["abc"]}}}
+
+    const anotherSegmentDef = {name: "Another Segment", description: "I did it again!", table_id: 1, show_in_getting_started: true,
+                               definition:{database: 1, query: {filter: ["def"]}}}
+    
+    const segmentCardDef = { name :"A card", display: "scalar", 
+                      dataset_query: {database: 1, table_id: 1, type: "query", query: {source_table: 1, "aggregation": ["count"], "filter": ["segment", 1]}},
+                      visualization_settings: {}}
+
+    // Scaffolding
+    beforeAll(async () => {
+        await login();
+
+    })
+
+
+
+    
+    describe("The Segments section of the Data Reference", async ()=>{
+
+        describe("Empty State", async () => {
+                it("Should show no segments in the list", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments");
+                mount(store.connectContainer(<SegmentListContainer />));
+                await store.waitForActions([FETCH_SEGMENTS])
+            })
+
+        });
+
+        describe("With Segments State", async () => {
+            var segmentIds = []
+
+            beforeAll(async () => {            
+                // Create some segments to have something to look at
+                var segment = await SegmentApi.create(segmentDef);
+                var anotherSegment = await SegmentApi.create(anotherSegmentDef);
+                segmentIds.push(segment.id)
+                segmentIds.push(anotherSegment.id)
+
+                })
+
+            afterAll(async () => {
+                // Delete the guide we created
+                // remove the metrics  we created   
+                // This is a bit messy as technically these are just archived
+                for (const id of segmentIds){
+                    await SegmentApi.delete({segmentId: id, revision_message: "Please"})
+                }
+            })
+
+
+            // segments list
+            it("Should show the segments in the list", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments");
+                mount(store.connectContainer(<SegmentListContainer />));
+                await store.waitForActions([FETCH_SEGMENTS])
+            })
+            // segment detail
+            it("Should show the segment detail view for a specific id", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]);
+                mount(store.connectContainer(<SegmentDetailContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE])
+            })
+
+            // segments field list
+            it("Should show the segment fields list", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]+"/fields");
+                mount(store.connectContainer(<SegmentFieldListContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE, FETCH_SEGMENT_FIELDS])
+            })
+            // segment detail
+            it("Should show the segment field detail view for a specific id", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]+"/fields/" + 1);
+                mount(store.connectContainer(<SegmentFieldDetailContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE, FETCH_SEGMENT_FIELDS])
+            })
+
+            // segment questions 
+            it("Should show no questions based on a new segment", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]+'/questions');
+                mount(store.connectContainer(<SegmentQuestionsContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE, LOAD_ENTITIES])
+            })
+            // segment revisions
+            it("Should show revisions", async () => {
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]+'/revisions');
+                mount(store.connectContainer(<SegmentRevisionsContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE, FETCH_SEGMENT_REVISIONS])
+            })
+
+
+
+            it("Should see a newly asked question in its questions list", async () => {
+                var card = await CardApi.create(segmentCardDef)
+
+                expect(card.name).toBe(segmentCardDef.name);
+                
+                await CardApi.delete({cardId: card.id})
+
+                const store = await createTestStore()    
+                store.pushPath("/reference/segments/"+segmentIds[0]+'/questions');
+                mount(store.connectContainer(<SegmentQuestionsContainer />));
+                await store.waitForActions([FETCH_SEGMENT_TABLE, LOAD_ENTITIES])
+            })
+                      
+        });
+    });
+
+
+
+});
\ No newline at end of file
diff --git a/frontend/src/metabase/reference/selectors.js b/frontend/src/metabase/reference/selectors.js
index e8bc19961dbdf7191258f23e9e654afc8cdedcad..7539d3c7ae23e80b5921010c47d26491e331b600 100644
--- a/frontend/src/metabase/reference/selectors.js
+++ b/frontend/src/metabase/reference/selectors.js
@@ -9,473 +9,21 @@ import {
 
 import {
     idsToObjectMap,
-    buildBreadcrumbs,
-    databaseToForeignKeys,
-    getQuestionUrl
+    databaseToForeignKeys
 } from "./utils";
 
 // import { getDatabases, getTables, getFields, getMetrics, getSegments } from "metabase/selectors/metadata";
 
-import { getShallowDatabases as getDatabases, getShallowTables as getTables, getShallowFields as getFields, getShallowMetrics as getMetrics, getShallowSegments as getSegments } from "metabase/selectors/metadata";
+import {
+    getShallowDatabases as getDatabases, getShallowTables as getTables, getShallowFields as getFields,
+    getShallowMetrics as getMetrics, getShallowSegments as getSegments
+} from "metabase/selectors/metadata";
 export { getShallowDatabases as getDatabases, getShallowTables as getTables, getShallowFields as getFields, getShallowMetrics as getMetrics, getShallowSegments as getSegments } from "metabase/selectors/metadata";
 
 import _ from "underscore";
 
-
-// there might be a better way to organize sections
-// it feels like I'm duplicating a lot of routing logic here
-//TODO: refactor to use different container components for each section
-// initialize section metadata in there
-// may not be worthwhile due to the extra boilerplate required
-// try using a higher-order component to reduce boilerplate?
-const referenceSections = {
-    [`/reference/guide`]: {
-        id: `/reference/guide`,
-        name: "Start here",
-        breadcrumb: "Guide",
-        fetch: {
-            fetchGuide: [],
-            fetchDashboards: [],
-            fetchMetrics: [],
-            fetchSegments: [],
-            fetchDatabasesWithMetadata: []
-        },
-        icon: "reference",
-        sidebar: false
-    },
-    [`/reference/metrics`]: {
-        id: `/reference/metrics`,
-        name: "Metrics",
-        empty: {
-            title: "Metrics are the official numbers that your team cares about",
-            adminMessage: "Defining common metrics for your team makes it even easier to ask questions",
-            message: "Metrics will appear here once your admins have created some",
-            image: "app/assets/img/metrics-list",
-            adminAction: "Learn how to create metrics",
-            adminLink: "http://www.metabase.com/docs/latest/administration-guide/06-segments-and-metrics.html"
-        },
-        breadcrumb: "Metrics",
-        // mapping of propname to args of dispatch function
-        fetch: {
-            fetchMetrics: [],
-            fetchSegments: []
-        },
-        get: 'getMetrics',
-        icon: "ruler"
-    },
-    [`/reference/segments`]: {
-        id: `/reference/segments`,
-        name: "Segments",
-        empty: {
-            title: "Segments are interesting subsets of tables",
-            adminMessage: "Defining common segments for your team makes it even easier to ask questions",
-            message: "Segments will appear here once your admins have created some",
-            image: "app/assets/img/segments-list",
-            adminAction: "Learn how to create segments",
-            adminLink: "http://www.metabase.com/docs/latest/administration-guide/06-segments-and-metrics.html"
-        },
-        breadcrumb: "Segments",
-        fetch: {
-            fetchMetrics: [],
-            fetchSegments: []
-        },
-        get: 'getSegments',
-        icon: "segment"
-    },
-    [`/reference/databases`]: {
-        id: `/reference/databases`,
-        name: "Databases and tables",
-        empty: {
-            title: "Metabase is no fun without any data",
-            adminMessage: "Your databses will appear here once you connect one",
-            message: "Databases will appear here once your admins have added some",
-            image: "app/assets/img/databases-list",
-            adminAction: "Connect a database",
-            adminLink: "/admin/databases/create"
-        },
-        breadcrumb: "Databases",
-        fetch: {
-            fetchMetrics: [],
-            fetchSegments: [],
-            fetchDatabases: []
-        },
-        get: 'getDatabases',
-        icon: "database",
-        itemIcon: "database"
-    }
-};
-
-const getReferenceSections = (state, props) => referenceSections;
-
-const getMetricSections = (metric, table, user) => metric ? {
-    [`/reference/metrics/${metric.id}`]: {
-        id: `/reference/metrics/${metric.id}`,
-        name: 'Details',
-        update: 'updateMetric',
-        type: 'metric',
-        breadcrumb: `${metric.name}`,
-        fetch: {
-            fetchMetricTable: [metric.id],
-            // currently the only way to fetch metrics important fields
-            fetchGuide: []
-        },
-        get: 'getMetric',
-        icon: "document",
-        headerIcon: "ruler",
-        headerLink: getQuestionUrl({
-            dbId: table && table.db_id,
-            tableId: metric.table_id,
-            metricId: metric.id
-        }),
-        parent: referenceSections[`/reference/metrics`]
-    },
-    [`/reference/metrics/${metric.id}/questions`]: {
-        id: `/reference/metrics/${metric.id}/questions`,
-        name: `Questions about ${metric.name}`,
-        empty: {
-            message: `Questions about this metric will appear here as they're added`,
-            icon: "all",
-            action: "Ask a question",
-            link: getQuestionUrl({
-                dbId: table && table.db_id,
-                tableId: metric.table_id,
-                metricId: metric.id
-            })
-        },
-        type: 'questions',
-        sidebar: 'Questions about this metric',
-        breadcrumb: `${metric.name}`,
-        fetch: {
-            fetchMetricTable: [metric.id],
-            fetchQuestions: []
-        },
-        get: 'getMetricQuestions',
-        icon: "all",
-        headerIcon: "ruler",
-        parent: referenceSections[`/reference/metrics`]
-    },
-    [`/reference/metrics/${metric.id}/revisions`]: {
-        id: `/reference/metrics/${metric.id}/revisions`,
-        name: `Revision history for ${metric.name}`,
-        sidebar: 'Revision history',
-        breadcrumb: `${metric.name}`,
-        hidden: user && !user.is_superuser,
-        fetch: {
-            fetchMetricRevisions: [metric.id]
-        },
-        get: 'getMetricRevisions',
-        icon: "history",
-        headerIcon: "ruler",
-        parent: referenceSections[`/reference/metrics`]
-    }
-} : {};
-
-const getSegmentSections = (segment, table, user) => segment ? {
-    [`/reference/segments/${segment.id}`]: {
-        id: `/reference/segments/${segment.id}`,
-        name: 'Details',
-        update: 'updateSegment',
-        type: 'segment',
-        questions: [
-            {
-                text: `Number of ${segment.name}`,
-                icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
-                link: getQuestionUrl({
-                    dbId: table && table.db_id,
-                    tableId: segment.table_id,
-                    segmentId: segment.id,
-                    getCount: true
-                })
-            },
-            {
-                text: `See all ${segment.name}`,
-                icon: "table2",
-                link: getQuestionUrl({
-                    dbId: table && table.db_id,
-                    tableId: segment.table_id,
-                    segmentId: segment.id
-                })
-            }
-        ],
-        breadcrumb: `${segment.name}`,
-        fetch: {
-            fetchSegmentTable: [segment.id]
-        },
-        get: 'getSegment',
-        icon: "document",
-        headerIcon: "segment",
-        headerLink: getQuestionUrl({
-            dbId: table && table.db_id,
-            tableId: segment.table_id,
-            segmentId: segment.id
-        }),
-        parent: referenceSections[`/reference/segments`]
-    },
-    [`/reference/segments/${segment.id}/fields`]: {
-        id: `/reference/segments/${segment.id}/fields`,
-        name: `Fields in ${segment.name}`,
-        empty: {
-            message: `Fields in this segment will appear here as they're added`,
-            icon: "fields"
-        },
-        sidebar: 'Fields in this segment',
-        fetch: {
-            fetchSegmentFields: [segment.id]
-        },
-        get: "getFieldsBySegment",
-        breadcrumb: `${segment.name}`,
-        icon: "fields",
-        headerIcon: "segment",
-        parent: referenceSections[`/reference/segments`]
-    },
-    [`/reference/segments/${segment.id}/questions`]: {
-        id: `/reference/segments/${segment.id}/questions`,
-        name: `Questions about ${segment.name}`,
-        empty: {
-            message: `Questions about this segment will appear here as they're added`,
-            icon: "all",
-            action: "Ask a question",
-            link: getQuestionUrl({
-                dbId: table && table.db_id,
-                tableId: segment.table_id,
-                segmentId: segment.id
-            })
-        },
-        type: 'questions',
-        sidebar: 'Questions about this segment',
-        breadcrumb: `${segment.name}`,
-        fetch: {
-            fetchSegmentTable: [segment.id],
-            fetchQuestions: []
-        },
-        get: 'getSegmentQuestions',
-        icon: "all",
-        headerIcon: "segment",
-        parent: referenceSections[`/reference/segments`]
-    },
-    [`/reference/segments/${segment.id}/revisions`]: {
-        id: `/reference/segments/${segment.id}/revisions`,
-        name: `Revision history for ${segment.name}`,
-        sidebar: 'Revision history',
-        breadcrumb: `${segment.name}`,
-        hidden: user && !user.is_superuser,
-        fetch: {
-            fetchSegmentRevisions: [segment.id]
-        },
-        get: 'getSegmentRevisions',
-        icon: "history",
-        headerIcon: "segment",
-        parent: referenceSections[`/reference/segments`]
-    }
-} : {};
-
-const getSegmentFieldSections = (segment, table, field, user) => segment && field ? {
-    [`/reference/segments/${segment.id}/fields/${field.id}`]: {
-        id: `/reference/segments/${segment.id}/fields/${field.id}`,
-        name: 'Details',
-        update: 'updateField',
-        type: 'field',
-        questions: [
-            {
-                text: `Number of ${table && table.display_name} grouped by ${field.display_name}`,
-                icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
-                link: getQuestionUrl({
-                    dbId: table && table.db_id,
-                    tableId: field.table_id,
-                    fieldId: field.id,
-                    getCount: true
-                })
-            },
-            {
-                text: `All distinct values of ${field.display_name}`,
-                icon: "table2",
-                link: getQuestionUrl({
-                    dbId: table && table.db_id,
-                    tableId: field.table_id,
-                    fieldId: field.id
-                })
-            }
-        ],
-        breadcrumb: `${field.display_name}`,
-        fetch: {
-            fetchSegmentFields: [segment.id]
-        },
-        get: "getFieldBySegment",
-        icon: "document",
-        headerIcon: "field",
-        parent: getSegmentSections(segment)[`/reference/segments/${segment.id}/fields`]
-    }
-} : {};
-
-const getDatabaseSections = (database) => database ? {
-    [`/reference/databases/${database.id}`]: {
-        id: `/reference/databases/${database.id}`,
-        name: 'Details',
-        update: 'updateDatabase',
-        type: 'database',
-        breadcrumb: `${database.name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id]
-        },
-        get: 'getDatabase',
-        icon: "document",
-        headerIcon: "database",
-        parent: referenceSections[`/reference/databases`]
-    },
-    [`/reference/databases/${database.id}/tables`]: {
-        id: `/reference/databases/${database.id}/tables`,
-        name: `Tables in ${database.name}`,
-        type: 'tables',
-        empty: {
-            message: `Tables in this database will appear here as they're added`,
-            icon: "table2"
-        },
-        sidebar: 'Tables in this database',
-        breadcrumb: `${database.name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id]
-        },
-        get: 'getTablesByDatabase',
-        icon: "table2",
-        headerIcon: "database",
-        parent: referenceSections[`/reference/databases`]
-    }
-} : {};
-
-const getTableSections = (database, table) => database && table ? {
-    [`/reference/databases/${database.id}/tables/${table.id}`]: {
-        id: `/reference/databases/${database.id}/tables/${table.id}`,
-        name: 'Details',
-        update: 'updateTable',
-        type: 'table',
-        questions: [
-            {
-                text: `Count of ${table.display_name}`,
-                icon: { name: "number", scale: 1, viewBox: "8 8 16 16" },
-                link: getQuestionUrl({
-                    dbId: table.db_id,
-                    tableId: table.id,
-                    getCount: true
-                })
-            },
-            {
-                text: `See raw data for ${table.display_name}`,
-                icon: "table2",
-                link: getQuestionUrl({
-                    dbId: table.db_id,
-                    tableId: table.id,
-                })
-            }
-        ],
-        breadcrumb: `${table.display_name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id]
-        },
-        get: 'getTable',
-        icon: "document",
-        headerIcon: "table2",
-        headerLink: getQuestionUrl({
-            dbId: table.db_id,
-            tableId: table.id,
-        }),
-        parent: getDatabaseSections(database)[`/reference/databases/${database.id}/tables`]
-    },
-    [`/reference/databases/${database.id}/tables/${table.id}/fields`]: {
-        id: `/reference/databases/${database.id}/tables/${table.id}/fields`,
-        name: `Fields in ${table.display_name}`,
-        empty: {
-            message: `Fields in this table will appear here as they're added`,
-            icon: "fields"
-        },
-        sidebar: 'Fields in this table',
-        breadcrumb: `${table.display_name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id]
-        },
-        get: "getFieldsByTable",
-        icon: "fields",
-        headerIcon: "table2",
-        parent: getDatabaseSections(database)[`/reference/databases/${database.id}/tables`]
-    },
-    [`/reference/databases/${database.id}/tables/${table.id}/questions`]: {
-        id: `/reference/databases/${database.id}/tables/${table.id}/questions`,
-        name: `Questions about ${table.display_name}`,
-        empty: {
-            message: `Questions about this table will appear here as they're added`,
-            icon: "all",
-            action: "Ask a question",
-            link: getQuestionUrl({
-                dbId: table.db_id,
-                tableId: table.id,
-            })
-        },
-        type: 'questions',
-        sidebar: 'Questions about this table',
-        breadcrumb: `${table.display_name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id], fetchQuestions: []
-        },
-        get: 'getTableQuestions',
-        icon: "all",
-        headerIcon: "table2",
-        parent: getDatabaseSections(database)[`/reference/databases/${database.id}/tables`]
-    }
-} : {};
-
-const getTableFieldSections = (database, table, field) => database && table && field ? {
-    [`/reference/databases/${database.id}/tables/${table.id}/fields/${field.id}`]: {
-        id: `/reference/databases/${database.id}/tables/${table.id}/fields/${field.id}`,
-        name: 'Details',
-        update: 'updateField',
-        type: 'field',
-        questions: [
-            {
-                text: `Number of ${table.display_name} grouped by ${field.display_name}`,
-                icon: { name: "bar", scale: 1, viewBox: "8 8 16 16" },
-                link: getQuestionUrl({
-                    dbId: database.id,
-                    tableId: table.id,
-                    fieldId: field.id,
-                    getCount: true,
-                    visualization: 'bar'
-                })
-            },
-            {
-                text: `Number of ${table.display_name} grouped by ${field.display_name}`,
-                icon: { name: "pie", scale: 1, viewBox: "8 8 16 16" },
-                link: getQuestionUrl({
-                    dbId: database.id,
-                    tableId: table.id,
-                    fieldId: field.id,
-                    getCount: true,
-                    visualization: 'pie'
-                })
-            },
-            {
-                text: `All distinct values of ${field.display_name}`,
-                icon: "table2",
-                link: getQuestionUrl({
-                    dbId: database.id,
-                    tableId: table.id,
-                    fieldId: field.id
-                })
-            }
-        ],
-        breadcrumb: `${field.display_name}`,
-        fetch: {
-            fetchDatabaseMetadata: [database.id]
-        },
-        get: "getField",
-        icon: "document",
-        headerIcon: "field",
-        parent: getTableSections(database, table)[`/reference/databases/${database.id}/tables/${table.id}/fields`]
-    }
-} : {};
-
 export const getUser = (state, props) => state.currentUser;
 
-export const getSectionId = (state, props) => props.location.pathname;
-
 export const getMetricId = (state, props) => Number.parseInt(props.params.metricId);
 export const getMetric = createSelector(
     [getMetricId, getMetrics],
@@ -490,19 +38,19 @@ export const getSegment = createSelector(
 
 export const getDatabaseId = (state, props) => Number.parseInt(props.params.databaseId);
 
-const getDatabase = createSelector(
+export const getDatabase = createSelector(
     [getDatabaseId, getDatabases],
     (databaseId, databases) => databases[databaseId] || { id: databaseId }
 );
 
 export const getTableId = (state, props) => Number.parseInt(props.params.tableId);
 // export const getTableId = (state, props) => Number.parseInt(props.params.tableId);
-const getTablesByDatabase = createSelector(
+export const getTablesByDatabase = createSelector(
     [getTables, getDatabase],
     (tables, database) => tables && database && database.tables ?
         idsToObjectMap(database.tables, tables) : {}
 );
-const getTableBySegment = createSelector(
+export const getTableBySegment = createSelector(
     [getSegment, getTables],
     (segment, tables) => segment && segment.table_id ? tables[segment.table_id] : {}
 );
@@ -519,26 +67,26 @@ export const getTable = createSelector(
 );
 
 export const getFieldId = (state, props) => Number.parseInt(props.params.fieldId);
-const getFieldsByTable = createSelector(
+export const getFieldsByTable = createSelector(
     [getTable, getFields],
     (table, fields) => table && table.fields ? idsToObjectMap(table.fields, fields) : {}
 );
-const getFieldsBySegment = createSelector(
+export const getFieldsBySegment = createSelector(
     [getTableBySegment, getFields],
     (table, fields) => table && table.fields ? idsToObjectMap(table.fields, fields) : {}
 );
-const getField = createSelector(
+export const getField = createSelector(
     [getFieldId, getFields],
     (fieldId, fields) => fields[fieldId] || { id: fieldId }
 );
-const getFieldBySegment = createSelector(
+export const getFieldBySegment = createSelector(
     [getFieldId, getFieldsBySegment],
     (fieldId, fields) => fields[fieldId] || { id: fieldId }
 );
 
 const getQuestions = (state, props) => getIn(state, ['questions', 'entities', 'cards']) || {};
 
-const getMetricQuestions = createSelector(
+export const getMetricQuestions = createSelector(
     [getMetricId, getQuestions],
     (metricId, questions) => Object.values(questions)
         .filter(question =>
@@ -552,17 +100,17 @@ const getMetricQuestions = createSelector(
 
 const getRevisions = (state, props) => state.metadata.revisions;
 
-const getMetricRevisions = createSelector(
+export const getMetricRevisions = createSelector(
     [getMetricId, getRevisions],
     (metricId, revisions) => getIn(revisions, ['metric', metricId]) || {}
 );
 
-const getSegmentRevisions = createSelector(
+export const getSegmentRevisions = createSelector(
     [getSegmentId, getRevisions],
     (segmentId, revisions) => getIn(revisions, ['segment', segmentId]) || {}
 );
 
-const getSegmentQuestions = createSelector(
+export const getSegmentQuestions = createSelector(
     [getSegmentId, getQuestions],
     (segmentId, questions) => Object.values(questions)
         .filter(question =>
@@ -573,7 +121,7 @@ const getSegmentQuestions = createSelector(
         .reduce((map, question) => assoc(map, question.id, question), {})
 );
 
-const getTableQuestions = createSelector(
+export const getTableQuestions = createSelector(
     [getTable, getQuestions],
     (table, questions) => Object.values(questions)
         .filter(question => question.table_id === table.id)
@@ -601,101 +149,10 @@ export const getForeignKeys = createSelector(
         foreignKeysBySegment : foreignKeysByDatabase
 )
 
-export const getSections = createSelector(
-    [getSectionId, getMetric, getSegment, getDatabase, getTable, getField, getFieldBySegment, getTableBySegment, getTableByMetric, getUser, getReferenceSections, getSegments, getMetrics],
-    (sectionId, metric, segment, database, table, field, fieldBySegment, tableBySegment, tableByMetric, user, referenceSections, segments, metrics) => {
-        // can be simplified if we had a single map of all sections
-        if (referenceSections[sectionId]) {
-            // filter out segments or metrics if we're not on that particular section and there are none
-            return _.omit(referenceSections,
-                Object.keys(metrics).length === 0 && sectionId !== "/reference/metrics" && "/reference/metrics",
-                Object.keys(segments).length === 0 && sectionId !== "/reference/segments"  && "/reference/segments",
-            );
-        }
-
-        const metricSections = getMetricSections(metric, tableByMetric, user);
-        if (metricSections[sectionId]) {
-            return metricSections;
-        }
-
-        const segmentSections = getSegmentSections(segment, tableBySegment, user);
-        if (segmentSections[sectionId]) {
-            return segmentSections;
-        }
-
-        const segmentFieldSections = getSegmentFieldSections(segment, tableBySegment, fieldBySegment);
-        if (segmentFieldSections[sectionId]) {
-            return segmentFieldSections;
-        }
-
-        const databaseSections = getDatabaseSections(database);
-        if (databaseSections[sectionId]) {
-            return databaseSections;
-        }
-
-        const tableSections = getTableSections(database, table);
-        if (tableSections[sectionId]) {
-            return tableSections;
-        }
-
-        const tableFieldSections = getTableFieldSections(database, table, field);
-        if (tableFieldSections[sectionId]) {
-            return tableFieldSections;
-        }
-
-        return {};
-    }
-);
-
-export const getSection = createSelector(
-    [getSectionId, getSections],
-    (sectionId, sections) => sections[sectionId] || {}
-);
-
-const dataSelectors = {
-    getMetric,
-    getMetricQuestions,
-    getMetricRevisions,
-    getMetrics,
-    getSegment,
-    getSegmentQuestions,
-    getSegmentRevisions,
-    getSegments,
-    getDatabase,
-    getDatabases,
-    getTable,
-    getTableQuestions,
-    getTables,
-    getTablesByDatabase,
-    getField,
-    getFieldBySegment,
-    getFields,
-    getFieldsByTable,
-    getFieldsBySegment
-};
-
-export const getData = (state, props) => {
-    const section = getSection(state, props);
-    if (!section) {
-        return {};
-    }
-    const selector = dataSelectors[section.get];
-    if (!selector) {
-        return {};
-    }
-
-    return selector(state, props);
-};
-
 export const getLoading = (state, props) => state.reference.isLoading;
 
 export const getError = (state, props) => state.reference.error;
 
-export const getBreadcrumbs = createSelector(
-    [getSection],
-    buildBreadcrumbs
-)
-
 export const getHasSingleSchema = createSelector(
     [getTablesByDatabase],
     (tables) => tables && Object.keys(tables).length > 0 ?
@@ -703,25 +160,6 @@ export const getHasSingleSchema = createSelector(
             .every((table, index, tables) => table.schema === tables[0].schema) : true
 )
 
-export const getHasDisplayName = createSelector(
-    [getSection],
-    (section) =>
-        section.type === 'table' ||
-        section.type === 'field'
-)
-
-export const getHasRevisionHistory = createSelector(
-    [getSection],
-    (section) =>
-        section.type === 'metric' ||
-        section.type === 'segment'
-)
-
-export const getHasQuestions = createSelector(
-    [getSection],
-    (section) => section.questions && section.questions.length > 0
-)
-
 export const getIsEditing = (state, props) => state.reference.isEditing;
 
 export const getIsFormulaExpanded = (state, props) => state.reference.isFormulaExpanded;
diff --git a/frontend/src/metabase/reference/utils.js b/frontend/src/metabase/reference/utils.js
index 0a85e12cc89dcc5ec8ef40b417179cc031cae011..1057f5c3ca4e9969150d37383ec30fcb82a72f0d 100644
--- a/frontend/src/metabase/reference/utils.js
+++ b/frontend/src/metabase/reference/utils.js
@@ -1,5 +1,4 @@
 import { assoc, assocIn, chain } from "icepick";
-import _ from "underscore";
 
 import { titleize, humanize } from "metabase/lib/formatting";
 import { startNewCard } from "metabase/lib/card";
@@ -13,344 +12,15 @@ export const idsToObjectMap = (ids, objects) => ids
     // hangs browser for large databases
     // .reduce((map, object) => assoc(map, object.id, object), {});
 
-const filterUntouchedFields = (fields, entity = {}) => Object.keys(fields)
+export const filterUntouchedFields = (fields, entity = {}) => Object.keys(fields)
     .filter(key =>
         fields[key] !== undefined &&
         entity[key] !== fields[key]
     )
     .reduce((map, key) => ({ ...map, [key]: fields[key] }), {});
 
-const isEmptyObject = (object) => Object.keys(object).length === 0;
+export const isEmptyObject = (object) => Object.keys(object).length === 0;
 
-export const tryFetchData = async (props) => {
-    const {
-        section,
-        clearError,
-        startLoading,
-        setError,
-        endLoading
-    } = props;
-
-    if (!(section && section.fetch)) {
-        return;
-    }
-
-    const fetch = section.fetch;
-    clearError();
-    startLoading();
-    try {
-        await Promise.all(Object.keys(fetch).map((fetchPropName) => {
-            const fetchData = props[fetchPropName];
-            const fetchArgs = fetch[fetchPropName] || [];
-            return fetchData(...fetchArgs);
-        }));
-    }
-    catch(error) {
-        setError(error);
-        console.error(error);
-    }
-
-    endLoading();
-}
-
-export const tryUpdateData = async (fields, props) => {
-    const {
-        entity,
-        guide,
-        section,
-        updateMetricImportantFields,
-        startLoading,
-        endLoading,
-        resetForm,
-        setError,
-        endEditing
-    } = props;
-
-    startLoading();
-    try {
-        const editedFields = filterUntouchedFields(fields, entity);
-        if (!isEmptyObject(editedFields)) {
-            const newEntity = {...entity, ...editedFields};
-            await props[section.update](newEntity);
-
-            if (section.type === 'metric' && fields.important_fields) {
-                const importantFieldIds = fields.important_fields.map(field => field.id);
-                const existingImportantFieldIds = guide.metric_important_fields && guide.metric_important_fields[entity.id];
-
-                const areFieldIdsIdentitical = existingImportantFieldIds &&
-                    existingImportantFieldIds.length === importantFieldIds.length &&
-                    existingImportantFieldIds.every(id => importantFieldIds.includes(id));
-
-                if (!areFieldIdsIdentitical) {
-                    await updateMetricImportantFields(entity.id, importantFieldIds);
-                    tryFetchData(props);
-                }
-            }
-        }
-    }
-    catch(error) {
-        setError(error);
-        console.error(error);
-    }
-
-    resetForm();
-    endLoading();
-    endEditing();
-}
-
-export const tryUpdateFields = async (formFields, props) => {
-    const {
-        entities,
-        updateField,
-        startLoading,
-        endLoading,
-        endEditing,
-        resetForm,
-        setError
-    } = props;
-
-    startLoading();
-    try {
-        const updatedFields = Object.keys(formFields)
-            .map(fieldId => ({
-                field: entities[fieldId],
-                formField: filterUntouchedFields(formFields[fieldId], entities[fieldId])
-            }))
-            .filter(({field, formField}) => !isEmptyObject(formField))
-            .map(({field, formField}) => ({...field, ...formField}));
-
-        await Promise.all(updatedFields.map(updateField));
-    }
-    catch(error) {
-        setError(error);
-        console.error(error);
-    }
-
-    resetForm();
-    endLoading();
-    endEditing();
-}
-
-export const tryUpdateGuide = async (formFields, props) => {
-    const {
-        guide,
-        dashboards,
-        metrics,
-        segments,
-        tables,
-        startLoading,
-        endLoading,
-        endEditing,
-        setError,
-        resetForm,
-        updateDashboard,
-        updateMetric,
-        updateSegment,
-        updateTable,
-        updateMetricImportantFields,
-        updateSetting,
-        fetchGuide,
-        clearRequestState
-    } = props;
-
-    startLoading();
-    try {
-        const updateNewEntities = ({
-            entities,
-            formFields,
-            updateEntity
-        }) => formFields.map(formField => {
-            if (!formField.id) {
-                return [];
-            }
-
-            const editedEntity = filterUntouchedFields(
-                assoc(formField, 'show_in_getting_started', true),
-                entities[formField.id]
-            );
-
-            if (isEmptyObject(editedEntity)) {
-                return [];
-            }
-
-            const newEntity = entities[formField.id];
-            const updatedNewEntity = {
-                ...newEntity,
-                ...editedEntity
-            };
-
-            const updatingNewEntity = updateEntity(updatedNewEntity);
-
-            return [updatingNewEntity];
-        });
-
-        const updateOldEntities = ({
-            newEntityIds,
-            oldEntityIds,
-            entities,
-            updateEntity
-        }) => oldEntityIds
-            .filter(oldEntityId => !newEntityIds.includes(oldEntityId))
-            .map(oldEntityId => {
-                const oldEntity = entities[oldEntityId];
-
-                const updatedOldEntity = assoc(
-                    oldEntity,
-                    'show_in_getting_started',
-                    false
-                );
-
-                const updatingOldEntity = updateEntity(updatedOldEntity);
-
-                return [updatingOldEntity];
-            });
-        //FIXME: necessary because revision_message is a mandatory field
-        // even though we don't actually keep track of changes to caveats/points_of_interest yet
-        const updateWithRevisionMessage = updateEntity => entity => updateEntity(assoc(
-            entity,
-            'revision_message',
-            'Updated in Getting Started guide.'
-        ));
-
-        const updatingDashboards = updateNewEntities({
-                entities: dashboards,
-                formFields: [formFields.most_important_dashboard],
-                updateEntity: updateDashboard
-            })
-            .concat(updateOldEntities({
-                newEntityIds: formFields.most_important_dashboard ?
-                    [formFields.most_important_dashboard.id] : [],
-                oldEntityIds: guide.most_important_dashboard ?
-                    [guide.most_important_dashboard] :
-                    [],
-                entities: dashboards,
-                updateEntity: updateDashboard
-            }));
-
-        const updatingMetrics = updateNewEntities({
-                entities: metrics,
-                formFields: formFields.important_metrics,
-                updateEntity: updateWithRevisionMessage(updateMetric)
-            })
-            .concat(updateOldEntities({
-                newEntityIds: formFields.important_metrics
-                    .map(formField => formField.id),
-                oldEntityIds: guide.important_metrics,
-                entities: metrics,
-                updateEntity: updateWithRevisionMessage(updateMetric)
-            }));
-
-        const updatingMetricImportantFields = formFields.important_metrics
-            .map(metricFormField => {
-                if (!metricFormField.id || !metricFormField.important_fields) {
-                    return [];
-                }
-                const importantFieldIds = metricFormField.important_fields
-                    .map(field => field.id);
-                const existingImportantFieldIds = guide.metric_important_fields[metricFormField.id];
-
-                const areFieldIdsIdentitical = existingImportantFieldIds &&
-                    existingImportantFieldIds.length === importantFieldIds.length &&
-                    existingImportantFieldIds.every(id => importantFieldIds.includes(id));
-                if (areFieldIdsIdentitical) {
-                    return [];
-                }
-
-                return [updateMetricImportantFields(metricFormField.id, importantFieldIds)];
-            });
-
-        const segmentFields = formFields.important_segments_and_tables
-            .filter(field => field.type === 'segment');
-
-        const updatingSegments = updateNewEntities({
-                entities: segments,
-                formFields: segmentFields,
-                updateEntity: updateWithRevisionMessage(updateSegment)
-            })
-            .concat(updateOldEntities({
-                newEntityIds: segmentFields
-                    .map(formField => formField.id),
-                oldEntityIds: guide.important_segments,
-                entities: segments,
-                updateEntity: updateWithRevisionMessage(updateSegment)
-            }));
-
-        const tableFields = formFields.important_segments_and_tables
-            .filter(field => field.type === 'table');
-
-        const updatingTables = updateNewEntities({
-                entities: tables,
-                formFields: tableFields,
-                updateEntity: updateTable
-            })
-            .concat(updateOldEntities({
-                newEntityIds: tableFields
-                    .map(formField => formField.id),
-                oldEntityIds: guide.important_tables,
-                entities: tables,
-                updateEntity: updateTable
-            }));
-
-        const updatingThingsToKnow = guide.things_to_know !== formFields.things_to_know ?
-            [updateSetting({key: 'getting-started-things-to-know', value: formFields.things_to_know })] :
-            [];
-
-        const updatingContactName = guide.contact && formFields.contact &&
-            guide.contact.name !== formFields.contact.name ?
-                [updateSetting({key: 'getting-started-contact-name', value: formFields.contact.name })] :
-                [];
-
-        const updatingContactEmail = guide.contact && formFields.contact &&
-            guide.contact.email !== formFields.contact.email ?
-                [updateSetting({key: 'getting-started-contact-email', value: formFields.contact.email })] :
-                [];
-
-        const updatingData = _.flatten([
-            updatingDashboards,
-            updatingMetrics,
-            updatingMetricImportantFields,
-            updatingSegments,
-            updatingTables,
-            updatingThingsToKnow,
-            updatingContactName,
-            updatingContactEmail
-        ]);
-
-        if (updatingData.length > 0) {
-            await Promise.all(updatingData);
-
-            clearRequestState({statePath: ['reference', 'guide']});
-
-            await fetchGuide();
-        }
-    }
-    catch(error) {
-        setError(error);
-        console.error(error);
-    }
-
-    resetForm();
-    endLoading();
-    endEditing();
-};
-
-const getBreadcrumb = (section, index, sections) => index !== sections.length - 1 ?
-    [section.breadcrumb, section.id] : [section.breadcrumb];
-
-const getParentSections = (section) => {
-    if (!section.parent) {
-        return [section];
-    }
-
-    const parentSections = []
-        .concat(getParentSections(section.parent), section);
-
-    return parentSections;
-};
-
-export const buildBreadcrumbs = (section) => getParentSections(section)
-    .map(getBreadcrumb)
-    .slice(-3);
 
 export const databaseToForeignKeys = (database) => database && database.tables_lookup ?
     Object.values(database.tables_lookup)
@@ -379,31 +49,9 @@ export const fieldsToFormFields = (fields) => Object.keys(fields)
     ])
     .reduce((array, keys) => array.concat(keys), []);
 
-export const separateTablesBySchema = (
-    tables,
-    section,
-    createSchemaSeparator,
-    createListItem
-) => Object.values(tables)
-    .sort((table1, table2) => table1.schema > table2.schema ? 1 :
-        table1.schema === table2.schema ? 0 : -1
-    )
-    .map((table, index, sortedTables) => {
-        if (!table || !table.id || !table.name) {
-            return;
-        }
-        // add schema header for first element and if schema is different from previous
-        const previousTableId = Object.keys(sortedTables)[index - 1];
-        return index === 0 ||
-            sortedTables[previousTableId].schema !== table.schema ?
-                [
-                    createSchemaSeparator(table),
-                    createListItem(table, index, section)
-                ] :
-                createListItem(table, index, section);
-    });
 
-export const getQuestion = ({dbId, tableId, fieldId, metricId, segmentId, getCount, visualization}) => {
+// TODO Atte Keinänen 7/3/17: Construct question with Question of metabase-lib instead of this using function
+export const getQuestion = ({dbId, tableId, fieldId, metricId, segmentId, getCount, visualization, metadata}) => {
     const newQuestion = startNewCard('query', dbId, tableId);
 
     // consider taking a look at Ramda as a possible underscore alternative?
@@ -416,7 +64,11 @@ export const getQuestion = ({dbId, tableId, fieldId, metricId, segmentId, getCou
         .updateIn(['display'], display => visualization || display)
         .updateIn(
             ['dataset_query', 'query', 'breakout'],
-            breakout => fieldId ? [fieldId] : breakout
+            (oldBreakout) => {
+                if (fieldId && metadata && metadata.fields[fieldId]) return [metadata.fields[fieldId].getDefaultBreakout()]
+                if (fieldId) return [fieldId];
+                return oldBreakout;
+            }
         )
         .value();
 
@@ -433,22 +85,6 @@ export const getQuestion = ({dbId, tableId, fieldId, metricId, segmentId, getCou
 
 export const getQuestionUrl = getQuestionArgs => Urls.question(null, getQuestion(getQuestionArgs));
 
-export const isGuideEmpty = ({
-    things_to_know,
-    contact,
-    most_important_dashboard,
-    important_metrics,
-    important_segments,
-    important_tables
-} = {}) => things_to_know ? false :
-    contact && contact.name ? false :
-    contact && contact.email ? false :
-    most_important_dashboard ? false :
-    important_metrics && important_metrics.length !== 0 ? false :
-    important_segments && important_segments.length !== 0 ? false :
-    important_tables && important_tables.length !== 0 ? false :
-    true;
-
 export const typeToLinkClass = {
     dashboard: 'text-green',
     metric: 'text-brand',
@@ -463,6 +99,7 @@ export const typeToBgClass = {
     table: 'bg-purple'
 };
 
+
 // little utility function to determine if we 'has' things, useful
 // for handling entity empty states
 export const has = (entity) => entity && entity.length > 0;
diff --git a/frontend/src/metabase/routes.jsx b/frontend/src/metabase/routes.jsx
index 2ef7fbfb05c7e85b604686a1e6d0bff60298d5c8..6314943f49b64346b3a88222333d9e3aa468a40a 100644
--- a/frontend/src/metabase/routes.jsx
+++ b/frontend/src/metabase/routes.jsx
@@ -54,12 +54,29 @@ import SettingsEditorApp from "metabase/admin/settings/containers/SettingsEditor
 import NotFound from "metabase/components/NotFound.jsx";
 import Unauthorized from "metabase/components/Unauthorized.jsx";
 
-import ReferenceApp from "metabase/reference/containers/ReferenceApp.jsx";
-import ReferenceEntity from "metabase/reference/containers/ReferenceEntity.jsx";
-import ReferenceEntityList from "metabase/reference/containers/ReferenceEntityList.jsx";
-import ReferenceFieldsList from "metabase/reference/containers/ReferenceFieldsList.jsx";
-import ReferenceRevisionsList from "metabase/reference/containers/ReferenceRevisionsList.jsx";
-import ReferenceGettingStartedGuide from "metabase/reference/containers/ReferenceGettingStartedGuide.jsx";
+// Reference Guide
+import GettingStartedGuideContainer from "metabase/reference/guide/GettingStartedGuideContainer.jsx";
+// Reference Metrics 
+import MetricListContainer from "metabase/reference/metrics/MetricListContainer.jsx";
+import MetricDetailContainer from "metabase/reference/metrics/MetricDetailContainer.jsx";
+import MetricQuestionsContainer from "metabase/reference/metrics/MetricQuestionsContainer.jsx";
+import MetricRevisionsContainer from "metabase/reference/metrics/MetricRevisionsContainer.jsx";
+// Reference Segments
+import SegmentListContainer from "metabase/reference/segments/SegmentListContainer.jsx";
+import SegmentDetailContainer from "metabase/reference/segments/SegmentDetailContainer.jsx";
+import SegmentQuestionsContainer from "metabase/reference/segments/SegmentQuestionsContainer.jsx";
+import SegmentRevisionsContainer from "metabase/reference/segments/SegmentRevisionsContainer.jsx";
+import SegmentFieldListContainer from "metabase/reference/segments/SegmentFieldListContainer.jsx";
+import SegmentFieldDetailContainer from "metabase/reference/segments/SegmentFieldDetailContainer.jsx";
+// Reference Databases
+import DatabaseListContainer from "metabase/reference/databases/DatabaseListContainer.jsx";
+import DatabaseDetailContainer from "metabase/reference/databases/DatabaseDetailContainer.jsx";
+import TableListContainer from "metabase/reference/databases/TableListContainer.jsx";
+import TableDetailContainer from "metabase/reference/databases/TableDetailContainer.jsx";
+import TableQuestionsContainer from "metabase/reference/databases/TableQuestionsContainer.jsx";
+import FieldListContainer from "metabase/reference/databases/FieldListContainer.jsx";
+import FieldDetailContainer from "metabase/reference/databases/FieldDetailContainer.jsx";
+
 
 import getAdminPermissionsRoutes from "metabase/admin/permissions/routes.jsx";
 
@@ -190,26 +207,26 @@ export const getRoutes = (store) =>
                 </Route>
 
                 {/* REFERENCE */}
-                <Route path="/reference" title="Data Reference" component={ReferenceApp}>
+                <Route path="/reference" title="Data Reference">
                     <IndexRedirect to="/reference/guide" />
-                    <Route path="guide" title="Getting Started" component={ReferenceGettingStartedGuide} />
-                    <Route path="metrics" component={ReferenceEntityList} />
-                    <Route path="metrics/:metricId" component={ReferenceEntity} />
-                    <Route path="metrics/:metricId/questions" component={ReferenceEntityList} />
-                    <Route path="metrics/:metricId/revisions" component={ReferenceRevisionsList} />
-                    <Route path="segments" component={ReferenceEntityList} />
-                    <Route path="segments/:segmentId" component={ReferenceEntity} />
-                    <Route path="segments/:segmentId/fields" component={ReferenceFieldsList} />
-                    <Route path="segments/:segmentId/fields/:fieldId" component={ReferenceEntity} />
-                    <Route path="segments/:segmentId/questions" component={ReferenceEntityList} />
-                    <Route path="segments/:segmentId/revisions" component={ReferenceRevisionsList} />
-                    <Route path="databases" component={ReferenceEntityList} />
-                    <Route path="databases/:databaseId" component={ReferenceEntity} />
-                    <Route path="databases/:databaseId/tables" component={ReferenceEntityList} />
-                    <Route path="databases/:databaseId/tables/:tableId" component={ReferenceEntity} />
-                    <Route path="databases/:databaseId/tables/:tableId/fields" component={ReferenceFieldsList} />
-                    <Route path="databases/:databaseId/tables/:tableId/fields/:fieldId" component={ReferenceEntity} />
-                    <Route path="databases/:databaseId/tables/:tableId/questions" component={ReferenceEntityList} />
+                    <Route path="guide" title="Getting Started" component={GettingStartedGuideContainer} />
+                    <Route path="metrics" component={MetricListContainer} />
+                    <Route path="metrics/:metricId" component={MetricDetailContainer} />
+                    <Route path="metrics/:metricId/questions" component={MetricQuestionsContainer} />
+                    <Route path="metrics/:metricId/revisions" component={MetricRevisionsContainer} />
+                    <Route path="segments" component={SegmentListContainer} />
+                    <Route path="segments/:segmentId" component={SegmentDetailContainer} />
+                    <Route path="segments/:segmentId/fields" component={SegmentFieldListContainer} />
+                    <Route path="segments/:segmentId/fields/:fieldId" component={SegmentFieldDetailContainer} />
+                    <Route path="segments/:segmentId/questions" component={SegmentQuestionsContainer} />
+                    <Route path="segments/:segmentId/revisions" component={SegmentRevisionsContainer} />
+                    <Route path="databases" component={DatabaseListContainer} />
+                    <Route path="databases/:databaseId" component={DatabaseDetailContainer} />
+                    <Route path="databases/:databaseId/tables" component={TableListContainer} />
+                    <Route path="databases/:databaseId/tables/:tableId" component={TableDetailContainer} />
+                    <Route path="databases/:databaseId/tables/:tableId/fields" component={FieldListContainer} />
+                    <Route path="databases/:databaseId/tables/:tableId/fields/:fieldId" component={FieldDetailContainer} />
+                    <Route path="databases/:databaseId/tables/:tableId/questions" component={TableQuestionsContainer} />
                 </Route>
 
                 {/* PULSE */}
@@ -259,6 +276,7 @@ export const getRoutes = (store) =>
                 <Route path="settings" title="Settings">
                     <IndexRedirect to="/admin/settings/setup" />
                     {/* <IndexRoute component={SettingsEditorApp} /> */}
+                    <Route path=":section/:authType" component={SettingsEditorApp} />
                     <Route path=":section" component={SettingsEditorApp} />
                 </Route>
 
diff --git a/frontend/src/metabase/services.js b/frontend/src/metabase/services.js
index 8035bd2b7e1b6c103fab0b9de13b703aa622083a..e3d2f855e37180b44bcab84e620d4b28a4d9afea 100644
--- a/frontend/src/metabase/services.js
+++ b/frontend/src/metabase/services.js
@@ -88,9 +88,13 @@ export const SlackApi = {
     updateSettings:              PUT("/api/slack/settings"),
 };
 
+export const LdapApi = {
+    updateSettings:              PUT("/api/ldap/settings")
+};
+
 export const MetabaseApi = {
     db_list:                     GET("/api/database"),
-    db_list_with_tables:         GET("/api/database?include_tables=true"),
+    db_list_with_tables:         GET("/api/database?include_tables=true&include_cards=true"),
     db_create:                  POST("/api/database"),
     db_add_sample_dataset:      POST("/api/database/sample_dataset"),
     db_get:                      GET("/api/database/:dbId"),
diff --git a/frontend/src/metabase/store.js b/frontend/src/metabase/store.js
index d326c6acf2648c1141c7222763350d20e158cee2..5b683f4396d38ff9955dc31e04f3100b787f89f4 100644
--- a/frontend/src/metabase/store.js
+++ b/frontend/src/metabase/store.js
@@ -21,7 +21,6 @@ const thunkWithDispatchAction = ({ dispatch, getState }) => next => action => {
 
         return action(dispatchAugmented, getState);
     }
-
     return next(action);
 };
 
@@ -32,7 +31,8 @@ if (DEBUG) {
 
 const devToolsExtension = window.devToolsExtension ? window.devToolsExtension() : (f => f);
 
-export function getStore(reducers, history, intialState) {
+export function getStore(reducers, history, intialState, enhancer = (a) => a) {
+
     const reducer = combineReducers({
         ...reducers,
         form,
@@ -43,6 +43,7 @@ export function getStore(reducers, history, intialState) {
 
     return createStore(reducer, intialState, compose(
         applyMiddleware(...middleware),
-        devToolsExtension
+        devToolsExtension,
+        enhancer,
     ));
-}
+}
\ No newline at end of file
diff --git a/frontend/src/metabase/user/components/UpdateUserDetails.jsx b/frontend/src/metabase/user/components/UpdateUserDetails.jsx
index acff1daecbc5e3fabb3eb72368626272fd42ce13..f58c3c5ec05fe8ff75a8278a1e7a0b1df6258b12 100644
--- a/frontend/src/metabase/user/components/UpdateUserDetails.jsx
+++ b/frontend/src/metabase/user/components/UpdateUserDetails.jsx
@@ -83,7 +83,7 @@ export default class UpdateUserDetails extends Component {
     render() {
         const { updateUserResult, user } = this.props;
         const { formError, valid } = this.state;
-        const managed = user.google_auth
+        const managed = user.google_auth || user.ldap_auth;
 
         return (
             <div>
@@ -101,7 +101,7 @@ export default class UpdateUserDetails extends Component {
                     </FormField>
 
                     <FormField fieldName="email" formError={formError}>
-                        <FormLabel title={ managed ? "Sign in with Google Email address" : "Email address"} fieldName="email" formError={formError} ></FormLabel>
+                        <FormLabel title={ user.google_auth ? "Sign in with Google Email address" : "Email address"} fieldName="email" formError={formError} ></FormLabel>
                         <input
                             ref="email"
                             className={
diff --git a/frontend/src/metabase/user/components/UserSettings.jsx b/frontend/src/metabase/user/components/UserSettings.jsx
index 036ff8898b190217e7a2ae22610ef2bd5533cada..ed1ade9083e471000ce283e301c0bad2b8784c38 100644
--- a/frontend/src/metabase/user/components/UserSettings.jsx
+++ b/frontend/src/metabase/user/components/UserSettings.jsx
@@ -31,7 +31,7 @@ export default class UserSettings extends Component {
 
     render() {
         let { tab } = this.props;
-        const nonSSOManagedAccount = !this.props.user.google_auth
+        const nonSSOManagedAccount = !this.props.user.google_auth && !this.props.user.ldap_auth;
 
         let allClasses = "Grid-cell md-no-flex md-mt1 text-brand-hover bordered border-brand-hover rounded p1 md-p3 block cursor-pointer text-centered md-text-left",
             tabClasses = {};
diff --git a/frontend/src/metabase/visualizations/components/TableInteractive.jsx b/frontend/src/metabase/visualizations/components/TableInteractive.jsx
index 8796ea2d4619982216240675d5c3069032badaa8..6db0134cef0b2341f46624c5a679cd4029c1c1c4 100644
--- a/frontend/src/metabase/visualizations/components/TableInteractive.jsx
+++ b/frontend/src/metabase/visualizations/components/TableInteractive.jsx
@@ -31,6 +31,7 @@ type Props = VisualizationProps & {
     height: number,
     sort: any,
     isPivoted: boolean,
+    onActionDismissal: () => void
 }
 type State = {
     columnWidths: number[],
@@ -362,7 +363,10 @@ export default class TableInteractive extends Component {
                         rowCount={rows.length}
                         rowHeight={ROW_HEIGHT}
                         cellRenderer={this.cellRenderer}
-                        onScroll={({ scrollLeft }) => onScroll({ scrollLeft })}
+                        onScroll={({ scrollLeft }) => {
+                            this.props.onActionDismissal()
+                            return onScroll({ scrollLeft })}
+                        }
                         scrollLeft={scrollLeft}
                         tabIndex={null}
                         overscanRowCount={20}
diff --git a/frontend/src/metabase/visualizations/components/Visualization.integ.spec.js b/frontend/src/metabase/visualizations/components/Visualization.integ.spec.js
index 9a97711b16624496ef9d0c2484a7aec904ce063e..ccb2854dfe941700a030f6e9f97928f0643e4f79 100644
--- a/frontend/src/metabase/visualizations/components/Visualization.integ.spec.js
+++ b/frontend/src/metabase/visualizations/components/Visualization.integ.spec.js
@@ -6,10 +6,8 @@ import { initializeQB, navigateToNewCardInsideQB } from "metabase/query_builder/
 import { parse as urlParse } from "url";
 
 import {
-    linkContainerToGlobalReduxStore,
     login,
-    globalReduxStore as store,
-    globalBrowserHistory as history
+    createTestStore
 } from "metabase/__support__/integrated_tests";
 
 import Question from "metabase-lib/lib/Question";
@@ -20,8 +18,10 @@ import {
 } from "metabase/__support__/sample_dataset_fixture";
 import ChartClickActions from "metabase/visualizations/components/ChartClickActions";
 
+const store = createTestStore()
+
 const getVisualization = (question, results, onChangeCardAndRun) =>
-    linkContainerToGlobalReduxStore(
+    store.connectContainer(
         <Visualization
             series={[{card: question.card(), data: results[0].data}]}
             onChangeCardAndRun={navigateToNewCardInsideQB}
diff --git a/frontend/src/metabase/visualizations/components/Visualization.jsx b/frontend/src/metabase/visualizations/components/Visualization.jsx
index 8d24e7694efbfa4a78bb30675e61e66b125cd9fe..2a14d3dc81b8585553313f353e4ebefa9f7ece22 100644
--- a/frontend/src/metabase/visualizations/components/Visualization.jsx
+++ b/frontend/src/metabase/visualizations/components/Visualization.jsx
@@ -29,11 +29,10 @@ import cx from "classnames";
 export const ERROR_MESSAGE_GENERIC = "There was a problem displaying this chart.";
 export const ERROR_MESSAGE_PERMISSION = "Sorry, you don't have permission to see this card."
 
-import type { VisualizationSettings} from "metabase/meta/types/Card";
-import type { HoverObject, ClickObject, Series } from "metabase/meta/types/Visualization";
-
-import Metadata from "metabase-lib/lib/metadata/Metadata";
 import Question from "metabase-lib/lib/Question";
+import type { Card as CardObject, VisualizationSettings } from "metabase/meta/types/Card";
+import type { HoverObject, ClickObject, Series, OnChangeCardAndRun } from "metabase/meta/types/Visualization";
+import Metadata from "metabase-lib/lib/metadata/Metadata";
 
 type Props = {
     series: Series,
@@ -63,7 +62,7 @@ type Props = {
 
     // for click actions
     metadata: Metadata,
-    onChangeCardAndRun: any => void,
+    onChangeCardAndRun: OnChangeCardAndRun,
 
     // used for showing content in place of visualization, e.x. dashcard filter mapping
     replacementContent: Element<any>,
@@ -239,11 +238,11 @@ export default class Visualization extends Component {
     };
 
     // Add the underlying card of current series to onChangeCardAndRun if available
-    handleOnChangeCardAndRun = ({ nextCard, seriesIndex }) => {
+    handleOnChangeCardAndRun = ({ nextCard, seriesIndex }: { nextCard: CardObject, seriesIndex: number }) => {
         const { series, clicked } = this.state;
 
         const index = seriesIndex || (clicked && clicked.seriesIndex) || 0;
-        const previousCard = series && series[index] && series[index].card;
+        const previousCard: ?CardObject = series && series[index] && series[index].card;
 
         this.props.onChangeCardAndRun({ nextCard, previousCard });
     }
@@ -256,6 +255,10 @@ export default class Visualization extends Component {
         this.setState({ error })
     }
 
+    hideActions = () => {
+        this.setState({ clicked: null })
+    }
+
     render() {
         const { actionButtons, className, showTitle, isDashboard, width, height, errorIcon, isSlow, expectedDuration, replacementContent } = this.props;
         const { series, CardVisualization } = this.state;
@@ -410,6 +413,7 @@ export default class Visualization extends Component {
                         visualizationIsClickable={this.visualizationIsClickable}
                         onRenderError={this.onRenderError}
                         onRender={this.onRender}
+                        onActionDismissal={this.hideActions}
                         gridSize={gridSize}
                         onChangeCardAndRun={this.props.onChangeCardAndRun ? this.handleOnChangeCardAndRun : null}
                     />
@@ -423,7 +427,7 @@ export default class Visualization extends Component {
                         clicked={clicked}
                         clickActions={clickActions}
                         onChangeCardAndRun={this.handleOnChangeCardAndRun}
-                        onClose={() => this.setState({ clicked: null })}
+                        onClose={this.hideActions}
                     />
                 }
             </div>
diff --git a/frontend/src/metabase/visualizations/components/settings/ChartSettingOrderedFields.jsx b/frontend/src/metabase/visualizations/components/settings/ChartSettingOrderedFields.jsx
index 4c9df5db752a42dd18ccd311e3c0ad5f014318c7..da61ad4455eab56327c50355187b36ae22a67fd5 100644
--- a/frontend/src/metabase/visualizations/components/settings/ChartSettingOrderedFields.jsx
+++ b/frontend/src/metabase/visualizations/components/settings/ChartSettingOrderedFields.jsx
@@ -56,7 +56,10 @@ export default class ChartSettingOrderedFields extends Component {
                         outline="list"
                     >
                         <div className={cx("flex align-center p1", { "text-grey-2": !item.enabled })} >
-                            <CheckBox checked={item.enabled} className={cx("text-brand", { "text-grey-2": !item.enabled })} onChange={(e) => this.setEnabled(i, e.target.checked)} invertChecked />
+                            <CheckBox
+                                checked={item.enabled}
+                                onChange={e => this.setEnabled(i, e.target.checked)}
+                            />
                             <span className="ml1 h4">{columnNames[item.name]}</span>
                             <Icon className="flex-align-right text-grey-2 mr1 cursor-pointer" name="grabber" width={14} height={14}/>
                         </div>
diff --git a/frontend/src/metabase/visualizations/lib/LineAreaBarRenderer.js b/frontend/src/metabase/visualizations/lib/LineAreaBarRenderer.js
index 93a4b857034a28a8d1ef5b47dcd69311f59a8520..5e338c432319e18697ed319ccbca2d7fdeb893e5 100644
--- a/frontend/src/metabase/visualizations/lib/LineAreaBarRenderer.js
+++ b/frontend/src/metabase/visualizations/lib/LineAreaBarRenderer.js
@@ -30,6 +30,7 @@ import {
 
 import { determineSeriesIndexFromElement } from "./tooltip";
 
+import { clipPathReference } from "metabase/lib/dom";
 import { formatValue, formatNumber } from "metabase/lib/formatting";
 import { parseTimestamp } from "metabase/lib/time";
 import { isStructured } from "metabase/meta/Card";
@@ -39,7 +40,7 @@ import { updateDateTimeFilter, updateNumericFilter } from "metabase/qb/lib/actio
 
 import { initBrush } from "./graph/brush";
 
-import type { SingleSeries, ClickObject } from "metabase/meta/types/Visualization"
+import type { VisualizationProps, SingleSeries, ClickObject } from "metabase/meta/types/Visualization"
 
 const MIN_PIXELS_PER_TICK = { x: 100, y: 32 };
 const BAR_PADDING_RATIO = 0.2;
@@ -588,7 +589,7 @@ function lineAndBarOnRender(chart, settings, onGoalHover, isSplitAxis, isStacked
                 .enter().append("svg:path")
                     .filter((d) => d != undefined)
                     .attr("d", (d) => "M" + d.join("L") + "Z")
-                    .attr("clip-path", (d,i) => "url(#clip-"+i+")")
+                    .attr("clip-path", (d,i) => clipPathReference("clip-" + i))
                     .on("mousemove", ({ point }) => {
                         let e = point[2];
                         dispatchUIEvent(e, "mousemove");
@@ -860,8 +861,13 @@ function forceSortedGroupsOfGroups(groupsOfGroups: CrossfilterGroup[][], indexMa
     }
 }
 
+type LineAreaBarProps = VisualizationProps & {
+    chartType: "line" | "area" | "bar" | "scatter",
+    isScalarSeries: boolean,
+    maxSeries: number
+}
 
-export default function lineAreaBar(element, {
+export default function lineAreaBar(element: Element, {
     series,
     onHoverChange,
     onVisualizationClick,
@@ -871,7 +877,7 @@ export default function lineAreaBar(element, {
     settings,
     maxSeries,
     onChangeCardAndRun
-}) {
+}: LineAreaBarProps) {
     const colors = settings["graph.colors"];
 
     // force histogram to be ordinal axis with zero-filled missing points
@@ -1038,6 +1044,7 @@ export default function lineAreaBar(element, {
                 }
             }
 
+            // $FlowFixMe
             series = series.map(s => updateIn(s, ["data", "cols", 1], (col) => ({
                 ...col,
                 display_name: "% " + getFriendlyName(col)
@@ -1104,9 +1111,9 @@ export default function lineAreaBar(element, {
             const card = series[0].card;
             const [start, end] = range;
             if (isDimensionTimeseries) {
-                onChangeCardAndRun({ nextCard: updateDateTimeFilter(card, column, start, end) });
+                onChangeCardAndRun({ nextCard: updateDateTimeFilter(card, column, start, end), previousCard: card });
             } else {
-                onChangeCardAndRun({ nextCard: updateNumericFilter(card, column, start, end) });
+                onChangeCardAndRun({ nextCard: updateNumericFilter(card, column, start, end), previousCard: card });
             }
         }
     }
diff --git a/frontend/src/metabase/visualizations/visualizations/Scalar.jsx b/frontend/src/metabase/visualizations/visualizations/Scalar.jsx
index 9608011fb23640ceefe4dbb0494bb6eaeaab9448..2ca4b525eb0b9d160c8dd037adbceaa8b51e3519 100644
--- a/frontend/src/metabase/visualizations/visualizations/Scalar.jsx
+++ b/frontend/src/metabase/visualizations/visualizations/Scalar.jsx
@@ -193,29 +193,31 @@ export default class Scalar extends Component {
                         {compactScalarValue}
                     </span>
                 </Ellipsified>
-                <div className={styles.Title + " flex align-center relative"}>
-                    <Ellipsified tooltip={card.name}>
-                        <span
-                            onClick={onChangeCardAndRun && (() => onChangeCardAndRun({ nextCard: card }))}
-                            className={cx("fullscreen-normal-text fullscreen-night-text", {
-                                "cursor-pointer": !!onChangeCardAndRun
-                            })}
-                        >
-                            {settings["card.title"]}
-                        </span>
-
-                    </Ellipsified>
-                    { description &&
-                        <div
-                            className="absolute top bottom hover-child flex align-center justify-center"
-                            style={{ right: -20, top: 2 }}
-                        >
-                          <Tooltip tooltip={description} maxWidth={'22em'}>
-                              <Icon name='infooutlined' />
-                          </Tooltip>
-                      </div>
-                    }
-                </div>
+                { this.props.isDashboard  && (
+                    <div className={styles.Title + " flex align-center relative"}>
+                        <Ellipsified tooltip={card.name}>
+                            <span
+                                onClick={onChangeCardAndRun && (() => onChangeCardAndRun({ nextCard: card }))}
+                                className={cx("fullscreen-normal-text fullscreen-night-text", {
+                                    "cursor-pointer": !!onChangeCardAndRun
+                                })}
+                            >
+                                <span className="Scalar-title">{settings["card.title"]}</span>
+                            </span>
+
+                        </Ellipsified>
+                        { description &&
+                            <div
+                                className="absolute top bottom hover-child flex align-center justify-center"
+                                style={{ right: -20, top: 2 }}
+                            >
+                              <Tooltip tooltip={description} maxWidth={'22em'}>
+                                  <Icon name='infooutlined' />
+                              </Tooltip>
+                          </div>
+                        }
+                    </div>
+                )}
             </div>
         );
     }
diff --git a/frontend/test/e2e/admin/datamodel.spec.js b/frontend/test/e2e/admin/datamodel.spec.js
index 10e9a67e6d43e50bfaaa5283fe42141da0899eb1..8908d894e583b9c8a83dc2c6b2541f6ea8524506 100644
--- a/frontend/test/e2e/admin/datamodel.spec.js
+++ b/frontend/test/e2e/admin/datamodel.spec.js
@@ -15,8 +15,9 @@ describeE2E("admin/datamodel", () => {
         ensureLoggedIn(server, driver, "bob@metabase.com", "12341234")
     );
 
+    // TODO Atte Keinänen 6/22/17: Data model specs are easy to convert to Enzyme, disabled until conversion has been done
     describe("data model editor", () => {
-        it("should allow admin to edit data model", async () => {
+        xit("should allow admin to edit data model", async () => {
             await driver.get(`${server.host}/admin/datamodel/database`);
 
             // hide orders table
@@ -45,7 +46,7 @@ describeE2E("admin/datamodel", () => {
             //TODO: verify tables and fields are hidden in query builder
         });
 
-        it("should allow admin to create segments and metrics", async () => {
+        xit("should allow admin to create segments and metrics", async () => {
             await driver.get(`${server.host}/admin/datamodel/database/1/table/2`);
 
             // add a segment
diff --git a/frontend/test/e2e/admin/settings.spec.js b/frontend/test/e2e/admin/settings.spec.js
index 8960e4f39f5d578856abec63158b5b747e7341f5..84226d9ed7ce3a2309e076a274632eccd74e2ecb 100644
--- a/frontend/test/e2e/admin/settings.spec.js
+++ b/frontend/test/e2e/admin/settings.spec.js
@@ -11,8 +11,9 @@ describeE2E("admin/settings", () => {
         ensureLoggedIn(server, driver, "bob@metabase.com", "12341234")
     );
 
+    // TODO Atte Keinänen 6/22/17: Disabled because we already have converted this to Jest&Enzyme in other branch
     describe("admin settings", () => {
-        it("should persist a setting", async () => {
+        xit("should persist a setting", async () => {
             // pick a random site name to try updating it to
             const siteName = "Metabase" + Math.random();
 
diff --git a/frontend/test/e2e/dashboard/dashboard.spec.js b/frontend/test/e2e/dashboard/dashboard.spec.js
index 7731cbcc500de8c617eee51e406313043e4da739..484e75e29aefb2ef0a5f61708a302137c64edfce 100644
--- a/frontend/test/e2e/dashboard/dashboard.spec.js
+++ b/frontend/test/e2e/dashboard/dashboard.spec.js
@@ -17,6 +17,7 @@ describeE2E("dashboards/dashboards", () => {
     });
 
     describe("dashboards list", () => {
+        // TODO Atte Keinänen 6/22/17: Failing test, disabled until converted to use Jest and Enzyme
         xit("should let you create new dashboards, see them, filter them and enter them", async () => {
             // Delegate dashboard creation to dashboard list test code
             await createDashboardInEmptyState();
diff --git a/frontend/test/e2e/dashboards/dashboards.spec.js b/frontend/test/e2e/dashboards/dashboards.spec.js
index e8d18e433cfaa43af3558349cdd998f6dce0ee06..58d26e4055f012c928ae9436f3182da1902bc9a1 100644
--- a/frontend/test/e2e/dashboards/dashboards.spec.js
+++ b/frontend/test/e2e/dashboards/dashboards.spec.js
@@ -17,6 +17,7 @@ describeE2E("dashboards/dashboards", () => {
             await ensureLoggedIn(server, driver, "bob@metabase.com", "12341234");
         });
 
+        // TODO Atte Keinänen 6/22/17: Failing test, disabled until converted to use Jest and Enzyme
         xit("should let you create new dashboards, see them, filter them and enter them", async () => {
             await d.get("/dashboards");
             await d.screenshot("screenshots/dashboards.png");
diff --git a/frontend/test/e2e/parameters/questions.spec.js b/frontend/test/e2e/parameters/questions.spec.js
index 969ff8f1625a2335e33a72f2af74ee3a8184893a..1ec91bcc28850e06a3e1d74e037925104ff6bd4a 100644
--- a/frontend/test/e2e/parameters/questions.spec.js
+++ b/frontend/test/e2e/parameters/questions.spec.js
@@ -56,7 +56,9 @@ describeE2E("parameters", () => {
             // make sure it's enabled
             await d.select(":react(SettingsSetting) .flex .text-bold").waitText("Enabled");
         });
-        it("should allow users to create parameterized SQL questions", async () => {
+
+        // TODO Atte Keinänen 6/22/17: Failing test, disabled until converted to use Jest and Enzyme
+        xit("should allow users to create parameterized SQL questions", async () => {
             await d::startNativeQuestion("select count(*) from products where {{category}}")
 
             await d.sleep(500);
diff --git a/frontend/test/e2e/query_builder/query_builder.spec.js b/frontend/test/e2e/query_builder/query_builder.spec.js
index 6209e3686b24b30b13a278f7a2fbe40658121869..44d3190311aa30dc7716594629d19bb04f151707 100644
--- a/frontend/test/e2e/query_builder/query_builder.spec.js
+++ b/frontend/test/e2e/query_builder/query_builder.spec.js
@@ -17,7 +17,8 @@ describeE2E("query_builder", () => {
     });
 
     describe("tables", () => {
-        it("should allow users to create pivot tables", async () => {
+        // TODO Atte Keinänen 6/22/17: Failing test, disabled until converted to use Jest and Enzyme
+        xit("should allow users to create pivot tables", async () => {
             // load the query builder and screenshot blank
             await d.get("/question");
             await d.screenshot("screenshots/qb-initial.png");
diff --git a/frontend/test/e2e/query_builder/tutorial.spec.js b/frontend/test/e2e/query_builder/tutorial.spec.js
index 9298adaf0a18211a64e7647e82286bf14ddf5504..5c8aae8c95d29a9bd684bf413f1c60d766de4b81 100644
--- a/frontend/test/e2e/query_builder/tutorial.spec.js
+++ b/frontend/test/e2e/query_builder/tutorial.spec.js
@@ -16,7 +16,8 @@ describeE2E("tutorial", () => {
         await ensureLoggedIn(server, driver, "bob@metabase.com", "12341234");
     });
 
-    it("should guide users through query builder tutorial", async () => {
+    // TODO Atte Keinänen 6/22/17: Failing test, disabled until converted to use Jest and Enzyme
+    xit("should guide users through query builder tutorial", async () => {
         await driver.get(`${server.host}/?new`);
         await waitForUrl(driver, `${server.host}/?new`);
 
diff --git a/frontend/test/run-integrated-tests.js b/frontend/test/run-integrated-tests.js
index d55451a3471230b469c02a38b742f6e543a4aa32..251b5629c16daf7193d2776e29108d1108285667 100755
--- a/frontend/test/run-integrated-tests.js
+++ b/frontend/test/run-integrated-tests.js
@@ -7,19 +7,49 @@ import { spawn } from "child_process";
 // use require for BackendResource to run it after the mock afterAll has been set
 const BackendResource = require("./e2e/support/backend.js").BackendResource
 const server = BackendResource.get({});
+const apiHost = process.env.E2E_HOST || server.host;
+
+const login = async () => {
+    const loginFetchOptions = {
+        method: "POST",
+        headers: new Headers({
+            "Accept": "application/json",
+            "Content-Type": "application/json"
+        }),
+        body: JSON.stringify({ username: "bob@metabase.com", password: "12341234"})
+    };
+    const result = await fetch(apiHost + "/api/session", loginFetchOptions);
+
+    let resultBody = null
+    try {
+        resultBody = await result.text();
+        resultBody = JSON.parse(resultBody);
+    } catch (e) {}
+
+    if (result.status >= 200 && result.status <= 299) {
+        console.log(`Successfully created a shared login with id ${resultBody.id}`)
+        return resultBody
+    } else {
+        const error = {status: result.status, data: resultBody }
+        console.log('A shared login attempt failed with the following error:');
+        console.log(error, {depth: null});
+        throw error
+    }
+}
 
 const init = async() => {
     await BackendResource.start(server)
+    const sharedLoginSession = await login()
 
-    const userArgs = process.argv.slice(2);
     const env = {
         ...process.env,
-        "E2E_HOST": process.env.E2E_HOST || server.host
+        "E2E_HOST": apiHost,
+        "SHARED_LOGIN_SESSION_ID": sharedLoginSession.id
     }
-
+    const userArgs = process.argv.slice(2);
     const jestProcess = spawn(
         "yarn",
-        ["run", "jest", "--", "--config", "jest.integ.conf.json", ...userArgs],
+        ["run", "jest", "--", "--maxWorkers=1", "--config", "jest.integ.conf.json", ...userArgs],
         {
             env,
             stdio: "inherit"
@@ -27,21 +57,21 @@ const init = async() => {
     );
 
     return new Promise((resolve, reject) => {
-        jestProcess.on('exit', () => { resolve(); })
+        jestProcess.on('exit', resolve)
     })
 }
 
-const cleanup = async () => {
+const cleanup = async (exitCode = 0) => {
     await jasmineAfterAllCleanup();
     await BackendResource.stop(server);
-    process.exit(0);
+    process.exit(exitCode);
 }
 
 init()
     .then(cleanup)
     .catch((e) => {
         console.error(e);
-        cleanup();
+        cleanup(1);
     });
 
 process.on('SIGTERM', () => {
diff --git a/frontend/test/unit/reference/utils.spec.js b/frontend/test/unit/reference/utils.spec.js
index ed576007bccc8269dfa1b6c922972d35fe9d8f5c..c96c4beb5971b808e3b689d1328d9e4914f511ff 100644
--- a/frontend/test/unit/reference/utils.spec.js
+++ b/frontend/test/unit/reference/utils.spec.js
@@ -1,148 +1,13 @@
 import {
-    tryFetchData,
-    tryUpdateData,
-    tryUpdateFields,
-    buildBreadcrumbs,
     databaseToForeignKeys,
-    separateTablesBySchema,
     getQuestion
 } from 'metabase/reference/utils';
 
+import { separateTablesBySchema } from "metabase/reference/databases/TableList"
 import { TYPE } from "metabase/lib/types";
 
 describe("Reference utils.js", () => {
-    const getProps = ({
-        section = {
-            fetch: {test1: [], test2: [2], test3: [3,4]}
-        },
-        entity = { foo: 'foo', bar: 'bar' },
-        entities = { foo: {foo: 'foo', bar: 'bar'}, bar: {foo: 'bar', bar: 'foo'} },
-        test1 = jasmine.createSpy('test1'),
-        test2 = jasmine.createSpy('test2'),
-        test3 = jasmine.createSpy('test3'),
-        updateField = jasmine.createSpy('updateField'),
-        clearError = jasmine.createSpy('clearError'),
-        resetForm = jasmine.createSpy('resetForm'),
-        endEditing = jasmine.createSpy('endEditing'),
-        startLoading = jasmine.createSpy('startLoading'),
-        setError = jasmine.createSpy('setError'),
-        endLoading = jasmine.createSpy('endLoading')
-    } = {}) => ({
-        section,
-        entity,
-        entities,
-        test1,
-        test2,
-        test3,
-        updateField,
-        clearError,
-        resetForm,
-        endEditing,
-        startLoading,
-        setError,
-        endLoading
-    });
-
-    describe("tryFetchData()", () => {
-        it("should call all fetch functions in section with correct arguments", async (done) => {
-            const props = getProps();
-            await tryFetchData(props);
-
-            expect(props.test1).toHaveBeenCalledWith();
-            expect(props.test2).toHaveBeenCalledWith(2);
-            expect(props.test3).toHaveBeenCalledWith(3, 4);
-            expect(props.clearError.calls.count()).toEqual(1);
-            expect(props.startLoading.calls.count()).toEqual(1);
-            expect(props.setError.calls.count()).toEqual(0);
-            expect(props.endLoading.calls.count()).toEqual(1);
-            done();
-        });
-
-        xit("should set error when error occurs", async () => {
-            const props = getProps(() => Promise.reject('test'));
-            tryFetchData(props).catch(error => console.error(error))
-
-            expect(props.test1).toHaveBeenCalledWith();
-            expect(props.test2).toHaveBeenCalledWith(2);
-            expect(props.test3).toHaveBeenCalledWith(3, 4);
-            expect(props.clearError.calls.count()).toEqual(1);
-            expect(props.startLoading.calls.count()).toEqual(1);
-            expect(props.setError.calls.count()).toEqual(0);
-            expect(props.endLoading.calls.count()).toEqual(1);
-        });
-    });
-
-    describe("tryUpdateData()", () => {
-        it("should call update function with merged entity", async (done) => {
-            const props = getProps({
-                section: {
-                    update: 'test1'
-                },
-                entity: { foo: 'foo', bar: 'bar' }
-            });
-            const fields = {bar: 'bar2'};
-
-            await tryUpdateData(fields, props);
-
-            expect(props.test1.calls.argsFor(0)[0]).toEqual({foo: 'foo', bar: 'bar2'});
-            expect(props.endEditing.calls.count()).toEqual(1);
-            expect(props.resetForm.calls.count()).toEqual(1);
-            expect(props.startLoading.calls.count()).toEqual(1);
-            expect(props.setError.calls.count()).toEqual(0);
-            expect(props.endLoading.calls.count()).toEqual(1);
-            done();
-        });
-
-        it("should ignore untouched fields when merging changed fields", async (done) => {
-            const props = getProps({
-                section: {
-                    update: 'test1'
-                },
-                entity: { foo: 'foo', bar: 'bar' }
-            });
-            const fields = {foo: '', bar: undefined, boo: 'boo'};
-
-            await tryUpdateData(fields, props);
-
-            expect(props.test1.calls.argsFor(0)[0]).toEqual({foo: '', bar: 'bar', boo: 'boo'});
-            expect(props.endEditing.calls.count()).toEqual(1);
-            expect(props.resetForm.calls.count()).toEqual(1);
-            expect(props.startLoading.calls.count()).toEqual(1);
-            expect(props.setError.calls.count()).toEqual(0);
-            expect(props.endLoading.calls.count()).toEqual(1);
-            done();
-        });
-    });
-
-    describe("tryUpdateFields()", () => {
-        it("should call update function with all updated fields", async (done) => {
-            const props = getProps();
-            const formFields = {
-                foo: {foo: undefined, bar: 'bar2'},
-                bar: {foo: '', bar: 'bar2'}
-            };
-
-            await tryUpdateFields(formFields, props);
-
-            expect(props.updateField.calls.argsFor(0)[0]).toEqual({foo: 'foo', bar: 'bar2'});
-            expect(props.updateField.calls.argsFor(1)[0]).toEqual({foo: '', bar: 'bar2'});
-            done();
-        });
-
-        it("should not call update function for items where all fields are untouched", async (done) => {
-            const props = getProps();
-            const formFields = {
-                foo: {foo: undefined, bar: undefined},
-                bar: {foo: undefined, bar: ''}
-            };
-
-            await tryUpdateFields(formFields, props);
 
-            expect(props.updateField.calls.argsFor(0)[0]).toEqual({foo: 'bar', bar: ''});
-            expect(props.updateField.calls.count()).toEqual(1);
-            done();
-        });
-    });
 
     describe("databaseToForeignKeys()", () => {
         it("should build foreignKey viewmodels from database", () => {
@@ -198,62 +63,7 @@ describe("Reference utils.js", () => {
         });
     });
 
-    describe("buildBreadcrumbs()", () => {
-        const section1 = {
-            id: 1,
-            breadcrumb: 'section1'
-        };
-
-        const section2 = {
-            id: 2,
-            breadcrumb: 'section2',
-            parent: section1
-        };
-
-        const section3 = {
-            id: 3,
-            breadcrumb: 'section3',
-            parent: section2
-        };
-
-        const section4 = {
-            id: 4,
-            breadcrumb: 'section4',
-            parent: section3
-        };
-
-        const section5 = {
-            id: 5,
-            breadcrumb: 'section5',
-            parent: section4
-        };
-
-        it("should build correct breadcrumbs from parent section", () => {
-            const breadcrumbs = buildBreadcrumbs(section1);
-            expect(breadcrumbs).toEqual([
-                [ 'section1' ]
-            ]);
-        });
-
-        it("should build correct breadcrumbs from child section", () => {
-            const breadcrumbs = buildBreadcrumbs(section3);
-            expect(breadcrumbs).toEqual([
-                [ 'section1', 1 ],
-                [ 'section2', 2 ],
-                [ 'section3' ]
-            ]);
-        });
-
-        it("should keep at most 3 highest level breadcrumbs", () => {
-            const breadcrumbs = buildBreadcrumbs(section5);
-            expect(breadcrumbs).toEqual([
-                [ 'section3', 3 ],
-                [ 'section4', 4 ],
-                [ 'section5' ]
-            ]);
-        });
-    });
-
+    
     describe("tablesToSchemaSeparatedTables()", () => {
         it("should add schema separator to appropriate locations", () => {
             const tables = {
@@ -265,14 +75,11 @@ describe("Reference utils.js", () => {
                 6: { id: 6, name: 'table6', schema: 'bar' }
             };
 
-            const section = {};
-
             const createSchemaSeparator = (table) => table.schema;
             const createListItem = (table) => table;
 
             const schemaSeparatedTables = separateTablesBySchema(
                 tables,
-                section,
                 createSchemaSeparator,
                 createListItem
             );
diff --git a/frontend/test/unit/visualizations/components/Visualization.spec.js b/frontend/test/unit/visualizations/components/Visualization.spec.js
index 459d760ce80d15a2497a5af140c2ed8fd9a5ec07..077a7b80fca0da4de530b5dbf8cbd0d31bfed248 100644
--- a/frontend/test/unit/visualizations/components/Visualization.spec.js
+++ b/frontend/test/unit/visualizations/components/Visualization.spec.js
@@ -1,6 +1,6 @@
 
 import React from "react";
-import { renderIntoDocument, scryRenderedComponentsWithType as scryWithType } from "react-dom/test-utils";
+import { renderIntoDocument, scryRenderedDOMComponentsWithClass, scryRenderedComponentsWithType as scryWithType } from "react-dom/test-utils";
 
 import Visualization from "metabase/visualizations/components/Visualization.jsx";
 
@@ -14,7 +14,7 @@ describe("Visualization", () => {
         describe("scalar card", () => {
             it("should not render title", () => {
                 let viz = renderVisualization({ series: [ScalarCard("Foo")] });
-                expect(getTitles(viz)).toEqual([]);
+                expect(getScalarTitles(viz)).toEqual([]);
             });
         });
 
@@ -38,9 +38,10 @@ describe("Visualization", () => {
 
     describe("in dashboard", () => {
         describe("scalar card", () => {
-            it("should not render title", () => {
-                let viz = renderVisualization({ series: [ScalarCard("Foo")], showTitle: true });
+            it("should render a scalar title, not a legend title", () => {
+                let viz = renderVisualization({ series: [ScalarCard("Foo")], showTitle: true, isDashboard: true });
                 expect(getTitles(viz)).toEqual([]);
+                expect(getScalarTitles(viz).length).toEqual(1);
             });
             it("should render title when loading", () => {
                 let viz = renderVisualization({ series: [ScalarCard("Foo", { data: null })], showTitle: true });
@@ -110,6 +111,10 @@ function renderVisualization(props) {
     return renderIntoDocument(<Visualization className="spread" {...props} />);
 }
 
+function getScalarTitles (scalarComponent) {
+    return scryRenderedDOMComponentsWithClass(scalarComponent, 'Scalar-title')
+}
+
 function getTitles(viz) {
     return scryWithType(viz, LegendHeader).map(header =>
         scryWithType(header, LegendItem).map(item =>
diff --git a/package.json b/package.json
index ee8bf91d43120f24a333cc27f08605f30f6488e2..eb8ac86c86cc2125611e8df0f8935e500af006eb 100644
--- a/package.json
+++ b/package.json
@@ -151,11 +151,11 @@
     "lint-eslint": "eslint --ext .js --ext .jsx --max-warnings 0 frontend/src frontend/test",
     "lint-prettier": "prettier --tab-width 4 -l 'frontend/src/metabase/{qb,new_question}/**/*.js*' 'frontend/src/metabase-lib/**/*.js' || (echo '\nThese files are not formatted correctly. Did you forget to run \"yarn run prettier\"?' && false)",
     "flow": "flow check",
-    "test": "yarn run test-jest && yarn run test-integrated && yarn run test-karma",
+    "test": "yarn run test-jest && yarn run test-karma",
     "test-karma": "karma start frontend/test/karma.conf.js --single-run",
     "test-karma-watch": "karma start frontend/test/karma.conf.js --auto-watch --reporters nyan",
-    "test-jest": "jest --config jest.unit.conf.json",
-    "test-jest-watch": "jest --config jest.unit.conf.json --watch",
+    "test-jest": "jest --maxWorkers=10 --config jest.unit.conf.json",
+    "test-jest-watch": "jest --maxWorkers=10 --config jest.unit.conf.json --watch",
     "test-integrated": "babel-node ./frontend/test/run-integrated-tests.js",
     "test-integrated-watch": "babel-node ./frontend/test/run-integrated-tests.js --watch",
     "test-e2e": "JASMINE_CONFIG_PATH=./frontend/test/e2e/support/jasmine.json jasmine",
@@ -180,4 +180,4 @@
       "git add"
     ]
   }
-}
+}
\ No newline at end of file
diff --git a/project.clj b/project.clj
index db982ea4083b96cf23faf6cec520072d08129564..504c2d9d5979520b628a92dab9694b26c470faf4 100644
--- a/project.clj
+++ b/project.clj
@@ -69,6 +69,7 @@
                  [net.sf.cssbox/cssbox "4.12"                         ; HTML / CSS rendering
                   :exclusions [org.slf4j/slf4j-api]]
                  [net.sourceforge.jtds/jtds "1.3.1"]                  ; Open Source SQL Server driver
+                 [org.clojars.pntblnk/clj-ldap "0.0.12"]              ; LDAP client
                  [org.liquibase/liquibase-core "3.5.3"]               ; migration management (Java lib)
                  [org.slf4j/slf4j-log4j12 "1.7.25"]                   ; abstraction for logging frameworks -- allows end user to plug in desired logging framework at deployment time
                  [org.yaml/snakeyaml "1.18"]                          ; YAML parser (required by liquibase)
diff --git a/resources/frontend_client/init.html b/resources/frontend_client/init.html
index 09137b50c19cd83b09c4a5b17b590fa1d9bb3ffa..5526fb2fd2af1166a617b2648f24effef51c3791 100644
--- a/resources/frontend_client/init.html
+++ b/resources/frontend_client/init.html
@@ -3,10 +3,30 @@
   <head>
     <meta charset="utf-8">
     <title>Metabase</title>
+
     <style media="screen">
       * {
         padding: 0;
         margin: 0;
+        font-family: Lato, 'Helvetica Neue', Helvetica, sans-serif;
+      }
+      progress[value] {
+        -webkit-appearance: none;
+        appearance: none;
+        width: 100%;
+        height: 4px;
+      }
+      progress[value]::-webkit-progress-bar {
+        background-color: #DBE9FA;
+        border-radius: 8px;
+      }
+      progress[value]::-webkit-progress-value {
+        background-color: #4A90E2;
+        border-radius: 8px;
+      }
+      #logo {
+        padding: 25px 0 20px 0;
+        margin-bottom: auto;
       }
       #container {
         position: absolute;
@@ -20,42 +40,133 @@
       #progress {
         margin-top: 75px;
       }
+      #heading {
+        font-size: 21px;
+        color: #616D75;
+        letter-spacing: 0;
+        line-height: 30px;
+        margin-top: 50px;
+        text-align: center;
+      }
       #status {
         margin-top: 15px;
         text-align: center;
-        font-family: Lato, 'Helvetica Neue', Helvetica, sans-serif;
-        font-size: 16px;
-        color: #509EE3;
+        font-size: 14px;
+        color: #93A1AB;
       }
-      progress[value] {
-        -webkit-appearance: none;
-        appearance: none;
-        width: 200px;
-        height: 5px;
+      #footer {
+        margin-top: auto;
+        text-align: center;
+        width: 100%;
+        padding-bottom: 20px;
       }
-      progress[value]::-webkit-progress-bar {
-        background-color: #DBE9FA;
-        border-radius: 8px;
+      #feature-image {
+        -webkit-transition: opacity 300ms cubic-bezier(.165,.84,.44,1);
+                transition: opacity 300ms cubic-bezier(.165,.84,.44,1);
       }
-      progress[value]::-webkit-progress-value {
-        background-color: #4A90E2;
-        border-radius: 8px;
+      .transparent {
+        opacity: 0;
+      }
+      .opaque {
+        opacity: 1;
+      }
+      .shadow {
+        box-shadow: 0 5px 25px 0 #BCC5CA;
+      }
+
+      .column-heading-img {
+        background-size: cover;
+        width: 504px;
+        height: 311px;
+        background-image:
+        url('');
+      }
+
+      .charts-img {
+        background-size: cover;
+        width: 547px;
+        height: 340px;
+        background-image:
+        url('');
+      }
+
+      .calendar-img {
+        background-size: cover;
+        width: 273px;
+        height: 321px;
+        background-image:
+        url('')
+      }
+
+      .drill-through-img {
+        width: 732px;
+        height: 266px;
+        background-size: cover;
+        background-image: url('');
+      }
+
+      .metabot-img {
+        width: 349px;
+        height: 280px;
+        min-height: 280px;
+        border-radius: 5px;
+        background-repeat: no-repeat;
+        background-image: url('');
       }
     </style>
   </head>
+
   <body>
     <div id="container">
-      <svg viewBox="0 0 66 85" width="94" height="116" fill="#509EE3">
-          <path d="M46.8253288,70.4935014 C49.5764899,70.4935014 51.8067467,68.1774705 51.8067467,65.3205017 C51.8067467,62.4635329 49.5764899,60.147502 46.8253288,60.147502 C44.0741676,60.147502 41.8439108,62.4635329 41.8439108,65.3205017 C41.8439108,68.1774705 44.0741676,70.4935014 46.8253288,70.4935014 Z M32.8773585,84.9779005 C35.6285197,84.9779005 37.8587764,82.6618697 37.8587764,79.8049008 C37.8587764,76.947932 35.6285197,74.6319011 32.8773585,74.6319011 C30.1261973,74.6319011 27.8959405,76.947932 27.8959405,79.8049008 C27.8959405,82.6618697 30.1261973,84.9779005 32.8773585,84.9779005 Z M32.8773585,70.4935014 C35.6285197,70.4935014 37.8587764,68.1774705 37.8587764,65.3205017 C37.8587764,62.4635329 35.6285197,60.147502 32.8773585,60.147502 C30.1261973,60.147502 27.8959405,62.4635329 27.8959405,65.3205017 C27.8959405,68.1774705 30.1261973,70.4935014 32.8773585,70.4935014 Z M18.9293882,70.4935014 C21.6805494,70.4935014 23.9108062,68.1774705 23.9108062,65.3205017 C23.9108062,62.4635329 21.6805494,60.147502 18.9293882,60.147502 C16.1782271,60.147502 13.9479703,62.4635329 13.9479703,65.3205017 C13.9479703,68.1774705 16.1782271,70.4935014 18.9293882,70.4935014 Z M46.8253288,56.0091023 C49.5764899,56.0091023 51.8067467,53.6930714 51.8067467,50.8361026 C51.8067467,47.9791337 49.5764899,45.6631029 46.8253288,45.6631029 C44.0741676,45.6631029 41.8439108,47.9791337 41.8439108,50.8361026 C41.8439108,53.6930714 44.0741676,56.0091023 46.8253288,56.0091023 Z M18.9293882,56.0091023 C21.6805494,56.0091023 23.9108062,53.6930714 23.9108062,50.8361026 C23.9108062,47.9791337 21.6805494,45.6631029 18.9293882,45.6631029 C16.1782271,45.6631029 13.9479703,47.9791337 13.9479703,50.8361026 C13.9479703,53.6930714 16.1782271,56.0091023 18.9293882,56.0091023 Z M46.8253288,26.8995984 C49.5764899,26.8995984 51.8067467,24.5835675 51.8067467,21.7265987 C51.8067467,18.8696299 49.5764899,16.553599 46.8253288,16.553599 C44.0741676,16.553599 41.8439108,18.8696299 41.8439108,21.7265987 C41.8439108,24.5835675 44.0741676,26.8995984 46.8253288,26.8995984 Z M32.8773585,41.5247031 C35.6285197,41.5247031 37.8587764,39.2086723 37.8587764,36.3517034 C37.8587764,33.4947346 35.6285197,31.1787037 32.8773585,31.1787037 C30.1261973,31.1787037 27.8959405,33.4947346 27.8959405,36.3517034 C27.8959405,39.2086723 30.1261973,41.5247031 32.8773585,41.5247031 Z M32.8773585,10.3459994 C35.6285197,10.3459994 37.8587764,8.02996853 37.8587764,5.17299969 C37.8587764,2.31603085 35.6285197,0 32.8773585,0 C30.1261973,0 27.8959405,2.31603085 27.8959405,5.17299969 C27.8959405,8.02996853 30.1261973,10.3459994 32.8773585,10.3459994 Z M32.8773585,26.8995984 C35.6285197,26.8995984 37.8587764,24.5835675 37.8587764,21.7265987 C37.8587764,18.8696299 35.6285197,16.553599 32.8773585,16.553599 C30.1261973,16.553599 27.8959405,18.8696299 27.8959405,21.7265987 C27.8959405,24.5835675 30.1261973,26.8995984 32.8773585,26.8995984 Z M18.9293882,26.8995984 C21.6805494,26.8995984 23.9108062,24.5835675 23.9108062,21.7265987 C23.9108062,18.8696299 21.6805494,16.553599 18.9293882,16.553599 C16.1782271,16.553599 13.9479703,18.8696299 13.9479703,21.7265987 C13.9479703,24.5835675 16.1782271,26.8995984 18.9293882,26.8995984 Z" opacity="0.2"></path>
-          <path d="M60.773299,70.4935014 C63.5244602,70.4935014 65.754717,68.1774705 65.754717,65.3205017 C65.754717,62.4635329 63.5244602,60.147502 60.773299,60.147502 C58.0221379,60.147502 55.7918811,62.4635329 55.7918811,65.3205017 C55.7918811,68.1774705 58.0221379,70.4935014 60.773299,70.4935014 Z M4.98141795,70.3527958 C7.73257912,70.3527958 9.96283591,68.0367649 9.96283591,65.1797961 C9.96283591,62.3228273 7.73257912,60.0067964 4.98141795,60.0067964 C2.23025679,60.0067964 0,62.3228273 0,65.1797961 C0,68.0367649 2.23025679,70.3527958 4.98141795,70.3527958 Z M60.773299,56.0091023 C63.5244602,56.0091023 65.754717,53.6930714 65.754717,50.8361026 C65.754717,47.9791337 63.5244602,45.6631029 60.773299,45.6631029 C58.0221379,45.6631029 55.7918811,47.9791337 55.7918811,50.8361026 C55.7918811,53.6930714 58.0221379,56.0091023 60.773299,56.0091023 Z M32.8773585,56.0091023 C35.6285197,56.0091023 37.8587764,53.6930714 37.8587764,50.8361026 C37.8587764,47.9791337 35.6285197,45.6631029 32.8773585,45.6631029 C30.1261973,45.6631029 27.8959405,47.9791337 27.8959405,50.8361026 C27.8959405,53.6930714 30.1261973,56.0091023 32.8773585,56.0091023 Z M4.98141795,55.8683967 C7.73257912,55.8683967 9.96283591,53.5523658 9.96283591,50.695397 C9.96283591,47.8384281 7.73257912,45.5223973 4.98141795,45.5223973 C2.23025679,45.5223973 0,47.8384281 0,50.695397 C0,53.5523658 2.23025679,55.8683967 4.98141795,55.8683967 Z M60.773299,41.5247031 C63.5244602,41.5247031 65.754717,39.2086723 65.754717,36.3517034 C65.754717,33.4947346 63.5244602,31.1787037 60.773299,31.1787037 C58.0221379,31.1787037 55.7918811,33.4947346 55.7918811,36.3517034 C55.7918811,39.2086723 58.0221379,41.5247031 60.773299,41.5247031 Z M46.8253288,41.5247031 C49.5764899,41.5247031 51.8067467,39.2086723 51.8067467,36.3517034 C51.8067467,33.4947346 49.5764899,31.1787037 46.8253288,31.1787037 C44.0741676,31.1787037 41.8439108,33.4947346 41.8439108,36.3517034 C41.8439108,39.2086723 44.0741676,41.5247031 46.8253288,41.5247031 Z M60.773299,26.8995984 C63.5244602,26.8995984 65.754717,24.5835675 65.754717,21.7265987 C65.754717,18.8696299 63.5244602,16.553599 60.773299,16.553599 C58.0221379,16.553599 55.7918811,18.8696299 55.7918811,21.7265987 C55.7918811,24.5835675 58.0221379,26.8995984 60.773299,26.8995984 Z M18.9293882,41.5247031 C21.6805494,41.5247031 23.9108062,39.2086723 23.9108062,36.3517034 C23.9108062,33.4947346 21.6805494,31.1787037 18.9293882,31.1787037 C16.1782271,31.1787037 13.9479703,33.4947346 13.9479703,36.3517034 C13.9479703,39.2086723 16.1782271,41.5247031 18.9293882,41.5247031 Z M4.98141795,41.3839975 C7.73257912,41.3839975 9.96283591,39.0679667 9.96283591,36.2109978 C9.96283591,33.354029 7.73257912,31.0379981 4.98141795,31.0379981 C2.23025679,31.0379981 0,33.354029 0,36.2109978 C0,39.0679667 2.23025679,41.3839975 4.98141795,41.3839975 Z M4.98141795,26.8995984 C7.73257912,26.8995984 9.96283591,24.5835675 9.96283591,21.7265987 C9.96283591,18.8696299 7.73257912,16.553599 4.98141795,16.553599 C2.23025679,16.553599 0,18.8696299 0,21.7265987 C0,24.5835675 2.23025679,26.8995984 4.98141795,26.8995984 Z"></path>
-      </svg>
-      <progress id="progress" max="100" value="0"></progress>
-      <div id="status">
-        Starting Metabase...
+      <div id="logo">
+        <svg viewBox="0 0 66 85" width="81.67" height="100" fill="#509EE3">
+            <path d="M46.8253288,70.4935014 C49.5764899,70.4935014 51.8067467,68.1774705 51.8067467,65.3205017 C51.8067467,62.4635329 49.5764899,60.147502 46.8253288,60.147502 C44.0741676,60.147502 41.8439108,62.4635329 41.8439108,65.3205017 C41.8439108,68.1774705 44.0741676,70.4935014 46.8253288,70.4935014 Z M32.8773585,84.9779005 C35.6285197,84.9779005 37.8587764,82.6618697 37.8587764,79.8049008 C37.8587764,76.947932 35.6285197,74.6319011 32.8773585,74.6319011 C30.1261973,74.6319011 27.8959405,76.947932 27.8959405,79.8049008 C27.8959405,82.6618697 30.1261973,84.9779005 32.8773585,84.9779005 Z M32.8773585,70.4935014 C35.6285197,70.4935014 37.8587764,68.1774705 37.8587764,65.3205017 C37.8587764,62.4635329 35.6285197,60.147502 32.8773585,60.147502 C30.1261973,60.147502 27.8959405,62.4635329 27.8959405,65.3205017 C27.8959405,68.1774705 30.1261973,70.4935014 32.8773585,70.4935014 Z M18.9293882,70.4935014 C21.6805494,70.4935014 23.9108062,68.1774705 23.9108062,65.3205017 C23.9108062,62.4635329 21.6805494,60.147502 18.9293882,60.147502 C16.1782271,60.147502 13.9479703,62.4635329 13.9479703,65.3205017 C13.9479703,68.1774705 16.1782271,70.4935014 18.9293882,70.4935014 Z M46.8253288,56.0091023 C49.5764899,56.0091023 51.8067467,53.6930714 51.8067467,50.8361026 C51.8067467,47.9791337 49.5764899,45.6631029 46.8253288,45.6631029 C44.0741676,45.6631029 41.8439108,47.9791337 41.8439108,50.8361026 C41.8439108,53.6930714 44.0741676,56.0091023 46.8253288,56.0091023 Z M18.9293882,56.0091023 C21.6805494,56.0091023 23.9108062,53.6930714 23.9108062,50.8361026 C23.9108062,47.9791337 21.6805494,45.6631029 18.9293882,45.6631029 C16.1782271,45.6631029 13.9479703,47.9791337 13.9479703,50.8361026 C13.9479703,53.6930714 16.1782271,56.0091023 18.9293882,56.0091023 Z M46.8253288,26.8995984 C49.5764899,26.8995984 51.8067467,24.5835675 51.8067467,21.7265987 C51.8067467,18.8696299 49.5764899,16.553599 46.8253288,16.553599 C44.0741676,16.553599 41.8439108,18.8696299 41.8439108,21.7265987 C41.8439108,24.5835675 44.0741676,26.8995984 46.8253288,26.8995984 Z M32.8773585,41.5247031 C35.6285197,41.5247031 37.8587764,39.2086723 37.8587764,36.3517034 C37.8587764,33.4947346 35.6285197,31.1787037 32.8773585,31.1787037 C30.1261973,31.1787037 27.8959405,33.4947346 27.8959405,36.3517034 C27.8959405,39.2086723 30.1261973,41.5247031 32.8773585,41.5247031 Z M32.8773585,10.3459994 C35.6285197,10.3459994 37.8587764,8.02996853 37.8587764,5.17299969 C37.8587764,2.31603085 35.6285197,0 32.8773585,0 C30.1261973,0 27.8959405,2.31603085 27.8959405,5.17299969 C27.8959405,8.02996853 30.1261973,10.3459994 32.8773585,10.3459994 Z M32.8773585,26.8995984 C35.6285197,26.8995984 37.8587764,24.5835675 37.8587764,21.7265987 C37.8587764,18.8696299 35.6285197,16.553599 32.8773585,16.553599 C30.1261973,16.553599 27.8959405,18.8696299 27.8959405,21.7265987 C27.8959405,24.5835675 30.1261973,26.8995984 32.8773585,26.8995984 Z M18.9293882,26.8995984 C21.6805494,26.8995984 23.9108062,24.5835675 23.9108062,21.7265987 C23.9108062,18.8696299 21.6805494,16.553599 18.9293882,16.553599 C16.1782271,16.553599 13.9479703,18.8696299 13.9479703,21.7265987 C13.9479703,24.5835675 16.1782271,26.8995984 18.9293882,26.8995984 Z" opacity="0.2"></path>
+            <path d="M60.773299,70.4935014 C63.5244602,70.4935014 65.754717,68.1774705 65.754717,65.3205017 C65.754717,62.4635329 63.5244602,60.147502 60.773299,60.147502 C58.0221379,60.147502 55.7918811,62.4635329 55.7918811,65.3205017 C55.7918811,68.1774705 58.0221379,70.4935014 60.773299,70.4935014 Z M4.98141795,70.3527958 C7.73257912,70.3527958 9.96283591,68.0367649 9.96283591,65.1797961 C9.96283591,62.3228273 7.73257912,60.0067964 4.98141795,60.0067964 C2.23025679,60.0067964 0,62.3228273 0,65.1797961 C0,68.0367649 2.23025679,70.3527958 4.98141795,70.3527958 Z M60.773299,56.0091023 C63.5244602,56.0091023 65.754717,53.6930714 65.754717,50.8361026 C65.754717,47.9791337 63.5244602,45.6631029 60.773299,45.6631029 C58.0221379,45.6631029 55.7918811,47.9791337 55.7918811,50.8361026 C55.7918811,53.6930714 58.0221379,56.0091023 60.773299,56.0091023 Z M32.8773585,56.0091023 C35.6285197,56.0091023 37.8587764,53.6930714 37.8587764,50.8361026 C37.8587764,47.9791337 35.6285197,45.6631029 32.8773585,45.6631029 C30.1261973,45.6631029 27.8959405,47.9791337 27.8959405,50.8361026 C27.8959405,53.6930714 30.1261973,56.0091023 32.8773585,56.0091023 Z M4.98141795,55.8683967 C7.73257912,55.8683967 9.96283591,53.5523658 9.96283591,50.695397 C9.96283591,47.8384281 7.73257912,45.5223973 4.98141795,45.5223973 C2.23025679,45.5223973 0,47.8384281 0,50.695397 C0,53.5523658 2.23025679,55.8683967 4.98141795,55.8683967 Z M60.773299,41.5247031 C63.5244602,41.5247031 65.754717,39.2086723 65.754717,36.3517034 C65.754717,33.4947346 63.5244602,31.1787037 60.773299,31.1787037 C58.0221379,31.1787037 55.7918811,33.4947346 55.7918811,36.3517034 C55.7918811,39.2086723 58.0221379,41.5247031 60.773299,41.5247031 Z M46.8253288,41.5247031 C49.5764899,41.5247031 51.8067467,39.2086723 51.8067467,36.3517034 C51.8067467,33.4947346 49.5764899,31.1787037 46.8253288,31.1787037 C44.0741676,31.1787037 41.8439108,33.4947346 41.8439108,36.3517034 C41.8439108,39.2086723 44.0741676,41.5247031 46.8253288,41.5247031 Z M60.773299,26.8995984 C63.5244602,26.8995984 65.754717,24.5835675 65.754717,21.7265987 C65.754717,18.8696299 63.5244602,16.553599 60.773299,16.553599 C58.0221379,16.553599 55.7918811,18.8696299 55.7918811,21.7265987 C55.7918811,24.5835675 58.0221379,26.8995984 60.773299,26.8995984 Z M18.9293882,41.5247031 C21.6805494,41.5247031 23.9108062,39.2086723 23.9108062,36.3517034 C23.9108062,33.4947346 21.6805494,31.1787037 18.9293882,31.1787037 C16.1782271,31.1787037 13.9479703,33.4947346 13.9479703,36.3517034 C13.9479703,39.2086723 16.1782271,41.5247031 18.9293882,41.5247031 Z M4.98141795,41.3839975 C7.73257912,41.3839975 9.96283591,39.0679667 9.96283591,36.2109978 C9.96283591,33.354029 7.73257912,31.0379981 4.98141795,31.0379981 C2.23025679,31.0379981 0,33.354029 0,36.2109978 C0,39.0679667 2.23025679,41.3839975 4.98141795,41.3839975 Z M4.98141795,26.8995984 C7.73257912,26.8995984 9.96283591,24.5835675 9.96283591,21.7265987 C9.96283591,18.8696299 7.73257912,16.553599 4.98141795,16.553599 C2.23025679,16.553599 0,18.8696299 0,21.7265987 C0,24.5835675 2.23025679,26.8995984 4.98141795,26.8995984 Z"></path>
+        </svg>
+      </div>
+
+      <div id="feature-image" class="drill-through-img">&nbsp;</div>
+      <p id="heading">Click on your charts to dive deeper.</p>
+
+      <div id="footer">
+        <progress id="progress" max="100" value="0"></progress>
+        <div id="status">
+          Starting Metabase...
+        </div>
       </div>
     </div>
 
     <script type="text/javascript">
+      var content =
+        [
+          ['drill-through-img', 'Click on your charts to dive deeper.'],
+          ['metabot-img shadow', 'Bring your charts and data into Slack.'],
+          ['calendar-img', 'Dashboard filters let you filter all your charts at once.'],
+          ['charts-img shadow', 'Easily create and share beautiful dashboards.'],
+          ['column-heading-img', 'Click on column headings in your tables to explore them.']
+        ];
+
+      var featureImage = document.getElementById("feature-image");
+      var heading = document.getElementById("heading");
+
+      var counter = 0;
+
+      function switcher() {
+        setInterval(function() {
+          counter++;
+          if (counter == content.length) counter = 0;
+          featureImage.className = featureImage.className.replace(" opaque", "") + " transparent";
+          heading.className = "transparent";
+
+          // Need to somehow wait here for a sec before fading things back in
+          setTimeout(function() {
+            featureImage.className = content[counter][0] + " opaque";
+            heading.innerHTML = content[counter][1];
+            heading.className = "opaque";
+          }, 200);
+        }, 4000);
+      }
+
       var messages = [
         "Polishing tables…",
         "Scaling scalars…",
@@ -90,6 +201,8 @@
         }
         req.send();
       }
+
+      switcher();
       poll();
     </script>
   </body>
diff --git a/resources/migrations/000_migrations.yaml b/resources/migrations/000_migrations.yaml
index 8632de63fab46d6ad27dc26a97aa3c1d5b2435ca..f03ee95b114c973620a012f076b8174b2b8ea359 100644
--- a/resources/migrations/000_migrations.yaml
+++ b/resources/migrations/000_migrations.yaml
@@ -3021,7 +3021,7 @@ databaseChangeLog:
             columnNames: group_id, object
         ############################################################ Tweaks to metabase_table ############################################################
         # Modify the length of metabase_table.schema from 256 -> 254
-        # It turns out MySQL InnoDB indecies have to be 767 bytes or less (at least for older versions of MySQL)
+        # It turns out MySQL InnoDB indices have to be 767 bytes or less (at least for older versions of MySQL)
         # and 'utf8' text columns can use up to 3 bytes per character in MySQL -- see http://stackoverflow.com/a/22515986/1198455
         # So 256 * 3 = 768 bytes (too large to index / add unique constraints)
         # Drop this to 254; 254 * 3 = 762, which should give us room to index it along with a 4-byte integer as well if need be
@@ -3323,7 +3323,7 @@ databaseChangeLog:
       id: 49
       author: camsaul
       changes:
-        ######################################## Card public_uuid & indecies ########################################
+        ######################################## Card public_uuid & indices ########################################
         - addColumn:
             tableName: report_card
             columns:
@@ -3346,7 +3346,7 @@ databaseChangeLog:
             columns:
               column:
                 name: public_uuid
-        ######################################## Dashboard public_uuid & indecies ########################################
+        ######################################## Dashboard public_uuid & indices ########################################
         - addColumn:
             tableName: report_dashboard
             columns:
@@ -3648,6 +3648,30 @@ databaseChangeLog:
                   name: dashboard_id
   - changeSet:
       id: 56
+      author: wwwiiilll
+      changes:
+        - addColumn:
+            tableName: core_user
+            columns:
+              - column:
+                  name: ldap_auth
+                  type: boolean
+                  defaultValueBoolean: false
+                  constraints:
+                    nullable: false
+  - changeSet:
+      id: 57
+      author: camsaul
+      changes:
+        - addColumn:
+            tableName: report_card
+            columns:
+              - column:
+                  name: result_metadata
+                  type: text
+                  remarks: 'Serialized JSON containing metadata about the result columns from running the query.'
+  - changeSet:
+      id: 58
       author: senior
       changes:
         - addColumn:
diff --git a/src/metabase/api/card.clj b/src/metabase/api/card.clj
index 53df71031d685b3f36db501ab2a73c826276eed1..69b37ea253e11aaca871e5d52cc6f7bf916d9e13 100644
--- a/src/metabase/api/card.clj
+++ b/src/metabase/api/card.clj
@@ -29,7 +29,9 @@
             [metabase.query-processor
              [interface :as qpi]
              [util :as qputil]]
-            [metabase.query-processor.middleware.cache :as cache]
+            [metabase.query-processor.middleware
+             [cache :as cache]
+             [results-metadata :as results-metadata]]
             [metabase.util.schema :as su]
             [ring.util.codec :as codec]
             [schema.core :as s]
@@ -206,14 +208,44 @@
 
 
 ;;; ------------------------------------------------------------ Saving Cards ------------------------------------------------------------
+
+;; When a new Card is saved, we wouldn't normally have the results metadata for it until the first time its query is ran.
+;; As a workaround, we'll calculate this metadata and return it with all query responses, and then let the frontend
+;; pass it back to us when saving or updating a Card.
+;; As a basic step to make sure the Metadata is valid we'll also pass a simple checksum and have the frontend pass it back to us.
+;; See the QP `results-metadata` middleware namespace for more details
+
+(s/defn ^:private ^:always-validate result-metadata-for-query :- results-metadata/ResultsMetadata
+  "Fetch the results metadata for a QUERY by running the query and seeing what the QP gives us in return.
+   This is obviously a bit wasteful so hopefully we can avoid having to do this."
+  [query]
+  (binding [qpi/*disable-qp-logging* true]
+    (get-in (qp/process-query query) [:data :results_metadata :columns])))
+
+(s/defn ^:private ^:always-validate result-metadata :- (s/maybe results-metadata/ResultsMetadata)
+  "Get the right results metadata for this CARD. We'll check to see whether the METADATA passed in seems valid;
+   otherwise we'll run the query ourselves to get the right values."
+  [query metadata checksum]
+  (let [valid-metadata? (and (results-metadata/valid-checksum? metadata checksum)
+                             (s/validate results-metadata/ResultsMetadata metadata))]
+    (log/info (str "Card results metadata passed in to API is " (cond
+                                                                  valid-metadata? "VALID. Thanks!"
+                                                                  metadata        "INVALID. Running query to fetch correct metadata."
+                                                                  :else           "MISSING. Running query to fetch correct metadata.")))
+    (if valid-metadata?
+      metadata
+      (result-metadata-for-query query))))
+
 (api/defendpoint POST "/"
   "Create a new `Card`."
-  [:as {{:keys [dataset_query description display name visualization_settings collection_id]} :body}]
+  [:as {{:keys [dataset_query description display name visualization_settings collection_id result_metadata metadata_checksum]} :body}]
   {name                   su/NonBlankString
    description            (s/maybe su/NonBlankString)
    display                su/NonBlankString
    visualization_settings su/Map
-   collection_id          (s/maybe su/IntGreaterThanZero)}
+   collection_id          (s/maybe su/IntGreaterThanZero)
+   result_metadata        (s/maybe results-metadata/ResultsMetadata)
+   metadata_checksum      (s/maybe su/NonBlankString)}
   ;; check that we have permissions to run the query that we're trying to save
   (api/check-403 (perms/set-has-full-permissions-for-set? @api/*current-user-permissions-set* (card/query-perms-set dataset_query :write)))
   ;; check that we have permissions for the collection we're trying to save this card to, if applicable
@@ -227,7 +259,8 @@
          :display                display
          :name                   name
          :visualization_settings visualization_settings
-         :collection_id          collection_id)
+         :collection_id          collection_id
+         :result_metadata        (result-metadata dataset_query result_metadata metadata_checksum))
        (events/publish-event! :card-create)))
 
 
@@ -275,6 +308,17 @@
     (api/check-embedding-enabled)
     (api/check-superuser)))
 
+
+(defn- result-metadata-for-updating
+  "If CARD's query is being updated, return the value that should be saved for `result_metadata`. This *should* be passed
+   in to the API; if so, verifiy that it was correct (the checksum is valid); if not, go fetch it.
+   If the query has not changed, this returns `nil`, which means the value won't get updated below."
+  [card query metadata checksum]
+  (when (and query
+             (not= query (:dataset_query card)))
+
+    (result-metadata query metadata checksum)))
+
 (defn- publish-card-update!
   "Publish an event appropriate for the update(s) done to this CARD (`:card-update`, or archiving/unarchiving events)."
   [card archived?]
@@ -290,7 +334,7 @@
 
 (api/defendpoint PUT "/:id"
   "Update a `Card`."
-  [id :as {{:keys [dataset_query description display name visualization_settings archived collection_id enable_embedding embedding_params], :as body} :body}]
+  [id :as {{:keys [dataset_query description display name visualization_settings archived collection_id enable_embedding embedding_params result_metadata metadata_checksum], :as body} :body}]
   {name                   (s/maybe su/NonBlankString)
    dataset_query          (s/maybe su/Map)
    display                (s/maybe su/NonBlankString)
@@ -299,19 +343,23 @@
    archived               (s/maybe s/Bool)
    enable_embedding       (s/maybe s/Bool)
    embedding_params       (s/maybe su/EmbeddingParams)
-   collection_id          (s/maybe su/IntGreaterThanZero)}
+   collection_id          (s/maybe su/IntGreaterThanZero)
+   result_metadata        (s/maybe results-metadata/ResultsMetadata)
+   metadata_checksum      (s/maybe su/NonBlankString)}
   (let [card (api/write-check Card id)]
     ;; Do various permissions checks
     (check-allowed-to-change-collection card collection_id)
     (check-allowed-to-modify-query card dataset_query)
     (check-allowed-to-unarchive card archived)
     (check-allowed-to-change-embedding card enable_embedding embedding_params)
-    ;; ok, now save the Card
-    (db/update! Card id
-      ;; `collection_id` and `description` can be `nil` (in order to unset them). Other values should only be modified if they're passed in as non-nil
-      (u/select-keys-when body
-        :present #{:collection_id :description}
-        :non-nil #{:dataset_query :display :name :visualization_settings :archived :enable_embedding :embedding_params}))
+    ;; make sure we have the correct `result_metadata`
+    (let [body (assoc body :result_metadata (result-metadata-for-updating card dataset_query result_metadata metadata_checksum))]
+      ;; ok, now save the Card
+      (db/update! Card id
+        ;; `collection_id` and `description` can be `nil` (in order to unset them). Other values should only be modified if they're passed in as non-nil
+        (u/select-keys-when body
+          :present #{:collection_id :description}
+          :non-nil #{:dataset_query :display :name :visualization_settings :archived :enable_embedding :embedding_params :result_metadata})))
     ;; Fetch the updated Card from the DB
     (let [card (Card id)]
       (publish-card-update! card archived)
diff --git a/src/metabase/api/common/internal.clj b/src/metabase/api/common/internal.clj
index 41861e3909c1b1be8703c86941fce2404126f68e..1a265950edd8b3cfebe6136c4df236f674e417a1 100644
--- a/src/metabase/api/common/internal.clj
+++ b/src/metabase/api/common/internal.clj
@@ -112,12 +112,10 @@
 
     (arg-type :id) -> :int"
   [arg]
-  (-> auto-parse-arg-name-patterns
-      ((fn [[[pattern type] & rest-patterns]]
-         (or (when (re-find pattern (name arg))
-               type)
-             (when rest-patterns
-               (recur rest-patterns)))))))
+  (some (fn [[pattern type]]
+          (when (re-find pattern (name arg))
+            type))
+        auto-parse-arg-name-patterns))
 
 
 ;;; ## TYPIFY-ROUTE
diff --git a/src/metabase/api/database.clj b/src/metabase/api/database.clj
index e6e3808ee17a9df05d497388f9de07d93ddcf9bd..6de82e7a9b365cffa78122d0155f63088f75990a 100644
--- a/src/metabase/api/database.clj
+++ b/src/metabase/api/database.clj
@@ -9,13 +9,17 @@
              [events :as events]
              [sample-data :as sample-data]
              [util :as u]]
-            [metabase.api.common :as api]
+            [metabase.api
+             [common :as api]
+             [table :as table-api]]
             [metabase.models
+             [card :refer [Card]]
              [database :as database :refer [Database protected-password]]
              [field :refer [Field]]
              [interface :as mi]
              [permissions :as perms]
              [table :refer [Table]]]
+            [metabase.query-processor.util :as qputil]
             [metabase.util.schema :as su]
             [schema.core :as s]
             [toucan
@@ -30,16 +34,18 @@
 
 ;;; ------------------------------------------------------------ GET /api/database ------------------------------------------------------------
 
-
 (defn- add-tables [dbs]
   (let [db-id->tables (group-by :db_id (filter mi/can-read? (db/select Table
-                                                                  :active true
-                                                                  :db_id  [:in (map :id dbs)]
-                                                                  {:order-by [[:%lower.display_name :asc]]})))]
+                                                              :active true
+                                                              :db_id  [:in (map :id dbs)]
+                                                              {:order-by [[:%lower.display_name :asc]]})))]
     (for [db dbs]
       (assoc db :tables (get db-id->tables (:id db) [])))))
 
-(defn- add-native-perms-info [dbs]
+(defn- add-native-perms-info
+  "For each database in DBS add a `:native_permissions` field describing the current user's permissions for running native (e.g. SQL) queries.
+   Will be one of `:write`, `:read`, or `:none`."
+  [dbs]
   (for [db dbs]
     (let [user-has-perms? (fn [path-fn] (perms/set-has-full-permissions? @api/*current-user-permissions-set* (path-fn (u/get-id db))))]
       (assoc db :native_permissions (cond
@@ -47,16 +53,89 @@
                                       (user-has-perms? perms/native-read-path)      :read
                                       :else                                         :none)))))
 
-(defn- dbs-list [include-tables?]
+(defn- card-database-supports-nested-queries? [{{database-id :database} :dataset_query, :as card}]
+  (when database-id
+    (when-let [driver (driver/database-id->driver database-id)]
+      (driver/driver-supports? driver :nested-queries)
+      (mi/can-read? card))))
+
+(defn- card-has-ambiguous-columns?
+  "We know a card has ambiguous columns if any of the columns that come back end in `_2` (etc.) because that's what
+   clojure.java.jdbc 'helpfully' does for us automatically.
+   Presence of ambiguous columns disqualifies a query for use as a source query because something like
+
+     SELECT name
+     FROM (
+       SELECT x.name, y.name
+       FROM x
+       LEFT JOIN y on x.id = y.id
+     )
+
+   would be ambiguous. Too many things break when attempting to use a query like this. In the future, this may be
+   supported, but it will likely require rewriting the source SQL query to add appropriate aliases (this is even
+   trickier if the source query uses `SELECT *`)."
+  [{result-metadata :result_metadata}]
+  (some (partial re-find #"_2$")
+        (map (comp name :name) result-metadata)))
+
+(defn- card-uses-unnestable-aggregation?
+  "Since cumulative count and cumulative sum aggregations are done in Clojure-land we can't use Cards that
+   use queries with those aggregations as source queries. This function determines whether CARD is using one
+   of those queries so we can filter it out in Clojure-land."
+  [{{{aggregations :aggregation} :query} :dataset_query}]
+  (when (seq aggregations)
+    (some (fn [[ag-type]]
+            (contains? #{:cum-count :cum-sum} (qputil/normalize-token ag-type)))
+          ;; if we were passed in old-style [ag] instead of [[ag1], [ag2]] convert to new-style so we can iterate over list of ags
+          (if-not (sequential? (first aggregations))
+            [aggregations]
+            aggregations))))
+
+(defn- source-query-cards
+  "Fetch the Cards that can be used as source queries (e.g. presented as virtual tables)."
+  []
+  (as-> (db/select [Card :name :description :database_id :dataset_query :id :collection_id :result_metadata]
+          :result_metadata [:not= nil]
+          {:order-by [[:%lower.name :asc]]}) <>
+    (filter card-database-supports-nested-queries? <>)
+    (remove card-uses-unnestable-aggregation? <>)
+    (remove card-has-ambiguous-columns? <>)
+    (hydrate <> :collection)))
+
+(defn- cards-virtual-tables
+  "Return a sequence of 'virtual' Table metadata for eligible Cards.
+   (This takes the Cards from `source-query-cards` and returns them in a format suitable for consumption by the Query Builder.)"
+  [& {:keys [include-fields?]}]
+  (for [card (source-query-cards)]
+    (table-api/card->virtual-table card :include-fields? include-fields?)))
+
+(defn- saved-cards-virtual-db-metadata [& {:keys [include-fields?]}]
+  (when-let [virtual-tables (seq (cards-virtual-tables :include-fields? include-fields?))]
+    {:name               "Saved Questions"
+     :id                 database/virtual-id
+     :features           #{:basic-aggregations}
+     :tables             virtual-tables
+     :is_saved_questions true}))
+
+;; "Virtual" tables for saved cards simulate the db->schema->table hierarchy by doing fake-db->collection->card
+(defn- add-virtual-tables-for-saved-cards [dbs]
+  (if-let [virtual-db-metadata (saved-cards-virtual-db-metadata)]
+    ;; only add the 'Saved Questions' DB if there are Cards that can be used
+    (conj (vec dbs) virtual-db-metadata)
+    dbs))
+
+(defn- dbs-list [include-tables? include-cards?]
   (when-let [dbs (seq (filter mi/can-read? (db/select Database {:order-by [:%lower.name]})))]
-    (add-native-perms-info (if-not include-tables?
-                             dbs
-                             (add-tables dbs)))))
+    (cond-> (add-native-perms-info dbs)
+      include-tables? add-tables
+      include-cards?  add-virtual-tables-for-saved-cards)))
 
 (api/defendpoint GET "/"
   "Fetch all `Databases`."
-  [include_tables]
-  (or (dbs-list include_tables)
+  [include_tables include_cards]
+  {include_tables (s/maybe su/BooleanString)
+   include_cards  (s/maybe su/BooleanString)}
+  (or (dbs-list include_tables include_cards)
       []))
 
 
@@ -70,6 +149,17 @@
 
 ;;; ------------------------------------------------------------ GET /api/database/:id/metadata ------------------------------------------------------------
 
+;; Since the normal `:id` param in the normal version of the endpoint will never match with negative numbers
+;; we'll create another endpoint to specifically match the ID of the 'virtual' database. The `defendpoint` macro
+;; requires either strings or vectors for the route so we'll have to use a vector and create a regex to only
+;; match the virtual ID (and nothing else).
+(api/defendpoint GET ["/:virtual-db/metadata" :virtual-db (re-pattern (str database/virtual-id))]
+  "Endpoint that provides metadata for the Saved Questions 'virtual' database. Used for fooling the frontend
+   and allowing it to treat the Saved Questions virtual DB just like any other database."
+  []
+  (saved-cards-virtual-db-metadata :include-fields? true))
+
+
 (defn- db-metadata [id]
   (-> (api/read-check Database id)
       (hydrate [:tables [:fields :target :values] :segments :metrics])
diff --git a/src/metabase/api/dataset.clj b/src/metabase/api/dataset.clj
index 913c177238aca7600ac3581214c5b264b6910bff..22e476c51aac6a9c5ca20aeb0c85e364ff1a5106 100644
--- a/src/metabase/api/dataset.clj
+++ b/src/metabase/api/dataset.clj
@@ -3,6 +3,7 @@
   (:require [cheshire.core :as json]
             [clojure.data.csv :as csv]
             [clojure.string :as str]
+            [clojure.tools.logging :as log]
             [compojure.core :refer [POST]]
             [dk.ative.docjure.spreadsheet :as spreadsheet]
             [metabase
@@ -12,7 +13,8 @@
             [metabase.api.common :as api]
             [metabase.api.common.internal :refer [route-fn-name]]
             [metabase.models
-             [database :refer [Database]]
+             [card :refer [Card]]
+             [database :as database :refer [Database]]
              [query :as query]]
             [metabase.query-processor.util :as qputil]
             [metabase.util.schema :as su]
@@ -37,14 +39,28 @@
 
 ;;; ------------------------------------------------------------ Running a Query Normally ------------------------------------------------------------
 
+(defn- query->source-card-id
+  "Return the ID of the Card used as the \"source\" query of this query, if applicable; otherwise return `nil`.
+   Used so `:card-id` context can be passed along with the query so Collections perms checking is done if appropriate."
+  [outer-query]
+  (let [source-table (qputil/get-in-normalized outer-query [:query :source-table])]
+    (when (string? source-table)
+      (when-let [[_ card-id-str] (re-matches #"^card__(\d+$)" source-table)]
+        (log/info (str "Source query for this query is Card " card-id-str))
+        (u/prog1 (Integer/parseInt card-id-str)
+          (api/read-check Card <>))))))
+
 (api/defendpoint POST "/"
   "Execute a query and retrieve the results in the usual format."
   [:as {{:keys [database], :as query} :body}]
   {database s/Int}
-  (api/read-check Database database)
+  ;; don't permissions check the 'database' if it's the virtual database. That database doesn't actually exist :-)
+  (when-not (= database database/virtual-id)
+    (api/read-check Database database))
   ;; add sensible constraints for results limits on our query
-  (let [query (assoc query :constraints default-query-constraints)]
-    (qp/process-query-and-save-execution! query {:executed-by api/*current-user-id*, :context :ad-hoc})))
+  (let [source-card-id (query->source-card-id query)]
+    (qp/process-query-and-save-execution! (assoc query :constraints default-query-constraints)
+      {:executed-by api/*current-user-id*, :context :ad-hoc, :card-id source-card-id, :nested? (boolean source-card-id)})))
 
 
 ;;; ------------------------------------------------------------ Downloading Query Results in Other Formats ------------------------------------------------------------
diff --git a/src/metabase/api/embed.clj b/src/metabase/api/embed.clj
index 23aff5a7924c4a1dda898c04c69cfdd66ab54e2d..f1b9d8baff37e3a346ba14f5ad87cfa343679810 100644
--- a/src/metabase/api/embed.clj
+++ b/src/metabase/api/embed.clj
@@ -108,13 +108,13 @@
   "Remove the `:parameters` for DASHBOARD-OR-CARD that listed as `disabled` or `locked` in the EMBEDDING-PARAMS whitelist,
    or not present in the whitelist. This is done so the frontend doesn't display widgets for params the user can't set."
   [dashboard-or-card, embedding-params :- su/EmbeddingParams]
-  (let [params-to-remove (into #{} (concat (for [[param status] embedding-params
-                                                 :when          (not= status "enabled")]
-                                             param)
-                                           (for [{slug :slug} (:parameters dashboard-or-card)
-                                                 :let         [param (keyword slug)]
-                                                 :when        (not (contains? embedding-params param))]
-                                             param)))]
+  (let [params-to-remove (set (concat (for [[param status] embedding-params
+                                            :when          (not= status "enabled")]
+                                        param)
+                                      (for [{slug :slug} (:parameters dashboard-or-card)
+                                            :let         [param (keyword slug)]
+                                            :when        (not (contains? embedding-params param))]
+                                        param)))]
     (update dashboard-or-card :parameters remove-params-in-set params-to-remove)))
 
 (defn- remove-token-parameters
diff --git a/src/metabase/api/field.clj b/src/metabase/api/field.clj
index e1e0949e8e5ec761c6326e3ccaa9be0e6c89d5db..d513a6d80618b605d56b423a8c2a47822634e829 100644
--- a/src/metabase/api/field.clj
+++ b/src/metabase/api/field.clj
@@ -64,6 +64,8 @@
     [[:count     (metadata/field-count field)]
      [:distincts (metadata/field-distinct-count field)]]))
 
+(def ^:private empty-field-values
+  {:values {} :human_readable_values {}})
 
 (api/defendpoint GET "/:id/values"
   "If `Field`'s special type derives from `type/Category`, or its base type is `type/Boolean`, return
@@ -71,9 +73,19 @@
   [id]
   (let [field (api/read-check Field id)]
     (if-not (field-should-have-field-values? field)
-      {:values {} :human_readable_values {}}
+      empty-field-values
       (create-field-values-if-needed! field))))
 
+;; match things like GET /field-literal%2Ccreated_at%2Ctype%2FDatetime/values
+;; (this is how things like [field-literal,created_at,type/Datetime] look when URL-encoded)
+(api/defendpoint GET "/field-literal%2C:field-name%2Ctype%2F:field-type/values"
+  "Implementation of the field values endpoint for fields in the Saved Questions 'virtual' DB.
+   This endpoint is just a convenience to simplify the frontend code. It just returns the standard
+   'empty' field values response."
+  ;; we don't actually care what field-name or field-type are, so they're ignored
+  [_ _]
+  empty-field-values)
+
 
 ;; TODO - not sure this is used anymore
 (api/defendpoint POST "/:id/value_map_update"
diff --git a/src/metabase/api/ldap.clj b/src/metabase/api/ldap.clj
new file mode 100644
index 0000000000000000000000000000000000000000..5ef5571dc71661c041c590e0e437b0d07908d74c
--- /dev/null
+++ b/src/metabase/api/ldap.clj
@@ -0,0 +1,111 @@
+(ns metabase.api.ldap
+  "/api/ldap endpoints"
+  (:require [clojure.tools.logging :as log]
+            [clojure.set :as set]
+            [compojure.core :refer [PUT]]
+            [metabase.api.common :refer :all]
+            [metabase.config :as config]
+            [metabase.integrations.ldap :as ldap]
+            [metabase.models.setting :as setting]
+            [metabase.util.schema :as su]))
+
+(def ^:private ^:const mb-settings->ldap-details
+  {:ldap-enabled             :enabled
+   :ldap-host                :host
+   :ldap-port                :port
+   :ldap-bind-dn             :bind-dn
+   :ldap-password            :password
+   :ldap-security            :security
+   :ldap-user-base           :user-base
+   :ldap-user-filter         :user-filter
+   :ldap-attribute-email     :attribute-email
+   :ldap-attribute-firstname :attribute-firstname
+   :ldap-attribute-lastname  :attribute-lastname
+   :ldap-group-sync          :group-sync
+   :ldap-group-base          :group-base})
+
+(defn- humanize-error-messages
+  "Convert raw error message responses from our LDAP tests into our normal api error response structure."
+  [{:keys [status message]}]
+  (when (not= :SUCCESS status)
+    (log/warn "Problem connecting to LDAP server:" message)
+    (let [conn-error     {:errors {:ldap-host "Wrong host or port"
+                                   :ldap-port "Wrong host or port"}}
+          security-error {:errors {:ldap-port     "Wrong port or security setting"
+                                   :ldap-security "Wrong port or security setting"}}
+          bind-dn-error  {:errors {:ldap-bind-dn "Wrong bind DN"}}
+          creds-error    {:errors {:ldap-bind-dn  "Wrong bind DN or password"
+                                   :ldap-password "Wrong bind DN or password"}}]
+      (condp re-matches message
+        #".*UnknownHostException.*"
+        conn-error
+
+        #".*ConnectException.*"
+        conn-error
+
+        #".*SocketException.*"
+        security-error
+
+        #".*SSLException.*"
+        security-error
+
+        #"^For input string.*"
+        {:errors {:ldap-host "Invalid hostname, do not add the 'ldap://' or 'ldaps://' prefix"}}
+
+        #".*password was incorrect.*"
+        {:errors {:ldap-password "Password was incorrect"}}
+
+        #"^Unable to bind as user.*"
+        bind-dn-error
+
+        #"^Unable to parse bind DN.*"
+        {:errors {:ldap-bind-dn "Invalid bind DN"}}
+
+        #".*AcceptSecurityContext error, data 525,.*"
+        bind-dn-error
+
+        #".*AcceptSecurityContext error, data 52e,.*"
+        creds-error
+
+        #".*AcceptSecurityContext error, data 532,.*"
+        {:errors {:ldap-password "Password is expired"}}
+
+        #".*AcceptSecurityContext error, data 533,.*"
+        {:errors {:ldap-bind-dn "Account is disabled"}}
+
+        #".*AcceptSecurityContext error, data 701,.*"
+        {:errors {:ldap-bind-dn "Account is expired"}}
+
+        #"^User search base does not exist .*"
+        {:errors {:ldap-user-base "User search base does not exist or is unreadable"}}
+
+        #"^Group search base does not exist .*"
+        {:errors {:ldap-group-base "Group search base does not exist or is unreadable"}}
+
+        ;; everything else :(
+        #"(?s).*"
+        {:message message}))))
+
+(defendpoint PUT "/settings"
+  "Update LDAP related settings. You must be a superuser to do this."
+  [:as {settings :body}]
+  {settings su/Map}
+  (check-superuser)
+  (let [ldap-settings (select-keys settings (keys mb-settings->ldap-details))
+        ldap-details  (-> (set/rename-keys ldap-settings mb-settings->ldap-details)
+                          (assoc :port
+                            (when (seq (:ldap-port settings))
+                              (Integer/parseInt (:ldap-port settings)))))
+        results       (if-not (:ldap-enabled settings)
+                        ;; when disabled just respond with a success message
+                        {:status :SUCCESS}
+                        ;; otherwise validate settings
+                        (ldap/test-ldap-connection ldap-details))]
+    (if (= :SUCCESS (:status results))
+      ;; test succeeded, save our settings
+      (setting/set-many! ldap-settings)
+      ;; test failed, return result message
+      {:status 500
+       :body   (humanize-error-messages results)})))
+
+(define-routes)
diff --git a/src/metabase/api/routes.clj b/src/metabase/api/routes.clj
index 78e16f295229f6e846b8a0fb1a97f1b0805395cf..90b1b20751da34b986d2c1ab3526e4239a6eb504 100644
--- a/src/metabase/api/routes.clj
+++ b/src/metabase/api/routes.clj
@@ -15,6 +15,7 @@
              [geojson :as geojson]
              [getting-started :as getting-started]
              [label :as label]
+             [ldap :as ldap]
              [metric :as metric]
              [notify :as notify]
              [permissions :as permissions]
@@ -63,6 +64,7 @@
   (context "/getting_started" [] (+auth getting-started/routes))
   (context "/geojson"         [] (+auth geojson/routes))
   (context "/label"           [] (+auth label/routes))
+  (context "/ldap"            [] (+auth ldap/routes))
   (context "/metric"          [] (+auth metric/routes))
   (context "/notify"          [] (+apikey notify/routes))
   (context "/permissions"     [] (+auth permissions/routes))
diff --git a/src/metabase/api/session.clj b/src/metabase/api/session.clj
index 8432ae542678b85c31ae247352b8add6e707af41..dfe3116f2e33ac4a45f23dd8b42f56eebcc9e68c 100644
--- a/src/metabase/api/session.clj
+++ b/src/metabase/api/session.clj
@@ -11,6 +11,7 @@
              [util :as u]]
             [metabase.api.common :as api]
             [metabase.email.messages :as email]
+            [metabase.integrations.ldap :as ldap]
             [metabase.models
              [session :refer [Session]]
              [setting :refer [defsetting]]
@@ -33,28 +34,42 @@
       :user_id (:id user))
     (events/publish-event! :user-login {:user_id (:id user), :session_id <>, :first_login (not (boolean (:last_login user)))})))
 
-
 ;;; ## API Endpoints
 
 (def ^:private login-throttlers
-  {:email      (throttle/make-throttler :email)
-   :ip-address (throttle/make-throttler :email, :attempts-threshold 50)}) ; IP Address doesn't have an actual UI field so just show error by email
+  {:username   (throttle/make-throttler :username)
+   :ip-address (throttle/make-throttler :username, :attempts-threshold 50)}) ; IP Address doesn't have an actual UI field so just show error by username
 
 (api/defendpoint POST "/"
   "Login."
-  [:as {{:keys [email password]} :body, remote-address :remote-addr}]
-  {email    su/Email
+  [:as {{:keys [username password]} :body, remote-address :remote-addr}]
+  {username su/NonBlankString
    password su/NonBlankString}
   (throttle/check (login-throttlers :ip-address) remote-address)
-  (throttle/check (login-throttlers :email)      email)
-  (let [user (db/select-one [User :id :password_salt :password :last_login], :email email, :is_active true)]
+  (throttle/check (login-throttlers :username)   username)
+  ;; Primitive "strategy implementation", should be reworked for modular providers in #3210
+  (or
+    ;; First try LDAP if it's enabled
+    (when (ldap/ldap-configured?)
+      (try
+        (when-let [user-info (ldap/find-user username)]
+          (if (ldap/verify-password user-info password)
+            {:id (create-session! (ldap/fetch-or-create-user! user-info password))}
+            ;; Since LDAP knows about the user, fail here to prevent the local strategy to be tried with a possibly outdated password
+            (throw (ex-info "Password did not match stored password." {:status-code 400
+                                                                       :errors      {:password "did not match stored password"}}))))
+        (catch com.unboundid.util.LDAPSDKException e
+          (log/error (u/format-color 'red "Problem connecting to LDAP server, will fallback to local authentication") (.getMessage e)))))
+
+    ;; Then try local authentication
+    (when-let [user (db/select-one [User :id :password_salt :password :last_login], :email username, :is_active true)]
+      (when (pass/verify-password password (:password_salt user) (:password user))
+        {:id (create-session! user)}))
+
+    ;; If nothing succeeded complain about it
     ;; Don't leak whether the account doesn't exist or the password was incorrect
-    (when-not (and user
-                   (pass/verify-password password (:password_salt user) (:password user)))
-      (throw (ex-info "Password did not match stored password." {:status-code 400
-                                                                 :errors      {:password "did not match stored password"}})))
-    {:id (create-session! user)}))
-
+    (throw (ex-info "Password did not match stored password." {:status-code 400
+                                                               :errors      {:password "did not match stored password"}}))))
 
 (api/defendpoint DELETE "/"
   "Logout."
@@ -192,7 +207,7 @@
   (throttle/check (login-throttlers :ip-address) remote-address)
   ;; Verify the token is valid with Google
   (let [{:keys [given_name family_name email]} (google-auth-token-info token)]
-    (log/info "Successfully authenicated Google Auth token for:" given_name family_name)
+    (log/info "Successfully authenticated Google Auth token for:" given_name family_name)
     (google-auth-fetch-or-create-user! given_name family_name email)))
 
 
diff --git a/src/metabase/api/setup.clj b/src/metabase/api/setup.clj
index 078f7a1615f7347d47f229bf80143bf558b7f27e..ebb8d6cf7039f5b9e6f06dc27ee5c9dd2f2e6dff 100644
--- a/src/metabase/api/setup.clj
+++ b/src/metabase/api/setup.clj
@@ -153,7 +153,7 @@
       :triggered   (>= num-cards 30)}
      {:title       "Create segments"
       :group       "Curate your data"
-      :description "Keep everyone on the same page by creating canonnical sets of filters anyone can use while asking questions."
+      :description "Keep everyone on the same page by creating canonical sets of filters anyone can use while asking questions."
       :link        "/admin/datamodel/database"
       :completed   has-segments?
       :triggered   (>= num-cards 30)}]))
diff --git a/src/metabase/api/table.clj b/src/metabase/api/table.clj
index b9fe63cecd45c845342ca673c74956c66264a2ce..3a91877caac7d3af9074325fba84542d066f4d4b 100644
--- a/src/metabase/api/table.clj
+++ b/src/metabase/api/table.clj
@@ -4,10 +4,13 @@
             [compojure.core :refer [GET PUT]]
             [medley.core :as m]
             [metabase
+             [driver :as driver]
              [sync-database :as sync-database]
              [util :as u]]
             [metabase.api.common :as api]
             [metabase.models
+             [card :refer [Card]]
+             [database :as database :refer [Database]]
              [field :refer [Field]]
              [interface :as mi]
              [table :as table :refer [Table]]]
@@ -138,29 +141,35 @@
 (def ^:private coordinate-dimension-indexes
   (create-dim-index-seq :type/Coordinate))
 
-(defn- assoc-dimension-options [resp]
-  (-> resp
-      (assoc :dimension_options dimension-options-for-response)
-      (update :fields (fn [fields]
-                        (for [{:keys [base_type special_type min_value max_value] :as field} fields]
-                          (assoc field
-                            :dimension_options
-                            (cond
+(defn- assoc-field-dimension-options [{:keys [base_type special_type min_value max_value] :as field}]
+  (assoc field
+    :dimension_options
+    (cond
 
-                              (isa? base_type :type/DateTime)
-                              datetime-dimension-indexes
+      (isa? base_type :type/DateTime)
+      datetime-dimension-indexes
 
-                              (and min_value max_value
-                                   (isa? special_type :type/Coordinate))
-                              coordinate-dimension-indexes
+      (and min_value max_value
+           (isa? special_type :type/Coordinate))
+      coordinate-dimension-indexes
 
-                              (and min_value max_value
-                                   (isa? base_type :type/Number)
-                                   (or (nil? special_type) (isa? special_type :type/Number)))
-                              numeric-dimension-indexes
+      (and min_value max_value
+           (isa? base_type :type/Number)
+           (or (nil? special_type) (isa? special_type :type/Number)))
+      numeric-dimension-indexes
 
-                              :else
-                              [])))))))
+      :else
+      [])))
+
+(defn- assoc-dimension-options [resp driver]
+  (if (and driver (contains? (driver/features driver) :binning))
+    (-> resp
+        (assoc :dimension_options dimension-options-for-response)
+        (update :fields #(mapv assoc-field-dimension-options %)))
+    (-> resp
+        (assoc :dimension_options [])
+        (update :fields (fn [fields]
+                          (mapv #(assoc % :dimension_options []) fields))))))
 
 (api/defendpoint GET "/:id/query_metadata"
   "Get metadata about a `Table` useful for running queries.
@@ -170,16 +179,56 @@
   will any of its corresponding values be returned. (This option is provided for use in the Admin Edit Metadata page)."
   [id include_sensitive_fields]
   {include_sensitive_fields (s/maybe su/BooleanString)}
-  (-> (api/read-check Table id)
-      (hydrate :db [:fields :target] :field_values :segments :metrics)
-      (m/dissoc-in [:db :details])
-      assoc-dimension-options
-      (update-in [:fields] (if (Boolean/parseBoolean include_sensitive_fields)
-                             ;; If someone passes include_sensitive_fields return hydrated :fields as-is
-                             identity
-                             ;; Otherwise filter out all :sensitive fields
-                             (partial filter (fn [{:keys [visibility_type]}]
-                                               (not= (keyword visibility_type) :sensitive)))))))
+  (let [table (api/read-check Table id)
+        driver (driver/engine->driver (db/select-one-field :engine Database :id (:db_id table)))]
+    (-> table
+        (hydrate :db [:fields :target] :field_values :segments :metrics)
+        (m/dissoc-in [:db :details])
+        (assoc-dimension-options driver)
+        (update-in [:fields] (if (Boolean/parseBoolean include_sensitive_fields)
+                               ;; If someone passes include_sensitive_fields return hydrated :fields as-is
+                               identity
+                               ;; Otherwise filter out all :sensitive fields
+                               (partial filter (fn [{:keys [visibility_type]}]
+                                                 (not= (keyword visibility_type) :sensitive))))))))
+
+(defn- card-result-metadata->virtual-fields
+  "Return a sequence of 'virtual' fields metadata for the 'virtual' table for a Card in the Saved Questions 'virtual' database."
+  [card-id metadata]
+  (for [col metadata]
+    (assoc col
+      :table_id     (str "card__" card-id)
+      :id           [:field-literal (:name col) (or (:base_type col) :type/*)]
+      ;; don't return :special_type if it's a PK or FK because it confuses the frontend since it can't actually be used that way IRL
+      :special_type (when-let [special-type (keyword (:special_type col))]
+                      (when-not (or (isa? special-type :type/PK)
+                                    (isa? special-type :type/FK))
+                        special-type)))))
+
+(defn card->virtual-table
+  "Return metadata for a 'virtual' table for a CARD in the Saved Questions 'virtual' database. Optionally include 'virtual' fields as well."
+  [card & {:keys [include-fields?]}]
+  ;; if collection isn't already hydrated then do so
+  (let [card (hydrate card :colllection)]
+    (cond-> {:id           (str "card__" (u/get-id card))
+             :db_id        database/virtual-id
+             :display_name (:name card)
+             :schema       (get-in card [:collection :name] "All questions")
+             :description  (:description card)}
+      include-fields? (assoc :fields (card-result-metadata->virtual-fields (u/get-id card) (:result_metadata card))))))
+
+(api/defendpoint GET "/card__:id/query_metadata"
+  "Return metadata for the 'virtual' table for a Card."
+  [id]
+  (-> (db/select-one [Card :id :dataset_query :result_metadata :name :description :collection_id], :id id)
+      api/read-check
+      (card->virtual-table :include-fields? true)))
+
+(api/defendpoint GET "/card__:id/fks"
+  "Return FK info for the 'virtual' table for a Card. This is always empty, so this endpoint
+   serves mainly as a placeholder to avoid having to change anything on the frontend."
+  []
+  []) ; return empty array
 
 
 (api/defendpoint GET "/:id/fks"
diff --git a/src/metabase/api/user.clj b/src/metabase/api/user.clj
index 8358b439365942f93d99e34d8119fa637e6e3672..73fcd511c230f21d8b2984725ee0c14437825c67 100644
--- a/src/metabase/api/user.clj
+++ b/src/metabase/api/user.clj
@@ -5,6 +5,7 @@
              [common :as api]
              [session :as session-api]]
             [metabase.email.messages :as email]
+            [metabase.integrations.ldap :as ldap]
             [metabase.models.user :as user :refer [User]]
             [metabase.util :as u]
             [metabase.util.schema :as su]
@@ -21,7 +22,7 @@
 (api/defendpoint GET "/"
   "Fetch a list of all active `Users` for the admin People page."
   []
-  (db/select [User :id :first_name :last_name :email :is_superuser :google_auth :last_login], :is_active true))
+  (db/select [User :id :first_name :last_name :email :is_superuser :google_auth :ldap_auth :last_login], :is_active true))
 
 (defn- reactivate-user! [existing-user first-name last-name]
   (when-not (:is_active existing-user)
@@ -32,7 +33,9 @@
       :is_superuser  false
       ;; if the user orignally logged in via Google Auth and it's no longer enabled, convert them into a regular user (see Issue #3323)
       :google_auth   (boolean (and (:google_auth existing-user)
-                                   (session-api/google-auth-client-id))))) ; if google-auth-client-id is set it means Google Auth is enabled
+                                   (session-api/google-auth-client-id))) ; if google-auth-client-id is set it means Google Auth is enabled
+      :ldap_auth     (boolean (and (:ldap_auth existing-user)
+                                   (ldap/ldap-configured?)))))
   ;; now return the existing user whether they were originally active or not
   (User (u/get-id existing-user)))
 
diff --git a/src/metabase/db.clj b/src/metabase/db.clj
index d122d4c44b10daa9b9869c53822249613da4db1b..5a33009f5329cd3e3a9f01bd597fba0ee5a016b2 100644
--- a/src/metabase/db.clj
+++ b/src/metabase/db.clj
@@ -34,7 +34,7 @@
            ;; File-based DB
            (let [db-file-name (config/config-str :mb-db-file)
                  db-file      (io/file db-file-name)
-                 options      ";AUTO_SERVER=TRUE;DB_CLOSE_DELAY=-1"]
+                 options      ";DB_CLOSE_DELAY=-1"]
              (apply str "file:" (if (.isAbsolute db-file)
                                   ;; when an absolute path is given for the db file then don't mess with it
                                   [db-file-name options]
diff --git a/src/metabase/driver.clj b/src/metabase/driver.clj
index 5f88f50cd51c706bf9503d41ad27adbaf0c0e3d6..ae8af15930d621d83489f1aa91a64aa04306185a 100644
--- a/src/metabase/driver.clj
+++ b/src/metabase/driver.clj
@@ -138,7 +138,8 @@
   *  `:expressions` - Does this driver support [expressions](https://github.com/metabase/metabase/wiki/Query-Language-'98#expressions) (e.g. adding the values of 2 columns together)?
   *  `:dynamic-schema` -  Does this Database have no fixed definitions of schemas? (e.g. Mongo)
   *  `:native-parameters` - Does the driver support parameter substitution on native queries?
-  *  `:expression-aggregations` - Does the driver support using expressions inside aggregations? e.g. something like \"sum(x) + count(y)\" or \"avg(x + y)\"")
+  *  `:expression-aggregations` - Does the driver support using expressions inside aggregations? e.g. something like \"sum(x) + count(y)\" or \"avg(x + y)\"
+  *  `:nested-queries` - Does the driver support using a query as the `:source-query` of another MBQL query? Examples are CTEs or subselects in SQL queries.")
 
   (field-values-lazy-seq ^clojure.lang.Sequential [this, ^FieldInstance field]
     "Return a lazy sequence of all values of FIELD.
@@ -283,7 +284,10 @@
   []
   (doseq [ns-symb @u/metabase-namespace-symbols
           :when   (re-matches #"^metabase\.driver\.[a-z0-9_]+$" (name ns-symb))]
-    (require ns-symb :reload)))
+    (require ns-symb)
+    (if-let [register-driver-fn (ns-resolve ns-symb (symbol "-init-driver"))]
+      (register-driver-fn)
+      (log/warn (format "No -init-driver function found for '%s'" (name ns-symb))))))
 
 (defn is-engine?
   "Is ENGINE a valid driver name?"
@@ -361,7 +365,8 @@
   (let [db-id->engine (memoize (fn [db-id] (db/select-one-field :engine Database, :id db-id)))]
     (fn [db-id]
       {:pre [db-id]}
-      (engine->driver (db-id->engine db-id)))))
+      (when-let [engine (db-id->engine db-id)]
+        (engine->driver engine)))))
 
 
 ;; ## Implementation-Agnostic Driver API
diff --git a/src/metabase/driver/bigquery.clj b/src/metabase/driver/bigquery.clj
index 3433b30a4fb797640912b08b6142a1b77c08da1c..8f1aa2622123737f56653a1f942441181e4b289d 100644
--- a/src/metabase/driver/bigquery.clj
+++ b/src/metabase/driver/bigquery.clj
@@ -210,15 +210,15 @@
 (defn- date-add [unit timestamp interval]
   (hsql/call :date_add timestamp interval (hx/literal unit)))
 
-;; µs = unix timestamp in microseconds. Most BigQuery functions like strftime require timestamps in this format
+;; microseconds = unix timestamp in microseconds. Most BigQuery functions like strftime require timestamps in this format
 
-(def ^:private ->µs (partial hsql/call :timestamp_to_usec))
+(def ^:private ->microseconds (partial hsql/call :timestamp_to_usec))
 
-(defn- µs->str [format-str µs]
+(defn- microseconds->str [format-str µs]
   (hsql/call :strftime_utc_usec µs (hx/literal format-str)))
 
 (defn- trunc-with-format [format-str timestamp]
-  (hx/->timestamp (µs->str format-str (->µs timestamp))))
+  (hx/->timestamp (microseconds->str format-str (->microseconds timestamp))))
 
 (defn- date [unit expr]
   {:pre [expr]}
@@ -503,7 +503,8 @@
           :features                 (constantly (set/union #{:basic-aggregations
                                                              :standard-deviation-aggregations
                                                              :native-parameters
-                                                             :expression-aggregations}
+                                                             :expression-aggregations
+                                                             :binning}
                                                            (when-not config/is-test?
                                                              ;; during unit tests don't treat bigquery as having FK support
                                                              #{:foreign-keys})))
@@ -511,4 +512,7 @@
           :format-custom-field-name (u/drop-first-arg format-custom-field-name)
           :mbql->native             (u/drop-first-arg mbql->native)}))
 
-(driver/register-driver! :bigquery driver)
+(defn -init-driver
+  "Register the BigQuery driver"
+  []
+  (driver/register-driver! :bigquery driver))
diff --git a/src/metabase/driver/crate.clj b/src/metabase/driver/crate.clj
index 1332754e049310b285a6c693e79ecaee5b8fc134..a322b2985e742f1fa63f4d4a26b189a2fd648eb3 100644
--- a/src/metabase/driver/crate.clj
+++ b/src/metabase/driver/crate.clj
@@ -117,5 +117,7 @@
           :unix-timestamp->timestamp crate-util/unix-timestamp->timestamp
           :current-datetime-fn       (constantly now)}))
 
-
-(driver/register-driver! :crate (CrateDriver.))
+(defn -init-driver
+  "Register the Crate driver"
+  []
+  (driver/register-driver! :crate (CrateDriver.)))
diff --git a/src/metabase/driver/druid.clj b/src/metabase/driver/druid.clj
index 26380b68295a4e401ed6319f944b3fac1c0b466c..1570bfeadba3d0dc9e81f23de74065a8bdf4d2ac 100644
--- a/src/metabase/driver/druid.clj
+++ b/src/metabase/driver/druid.clj
@@ -179,4 +179,7 @@
           :field-values-lazy-seq (u/drop-first-arg field-values-lazy-seq)
           :mbql->native          (u/drop-first-arg qp/mbql->native)}))
 
-(driver/register-driver! :druid (DruidDriver.))
+(defn -init-driver
+  "Register the druid driver1"
+  []
+  (driver/register-driver! :druid (DruidDriver.)))
diff --git a/src/metabase/driver/generic_sql.clj b/src/metabase/driver/generic_sql.clj
index a2157fa9076811d95f4403b76c97ff81cd1f7d8d..f7582549279341c33a8ad9afc8dccd990daa93a5 100644
--- a/src/metabase/driver/generic_sql.clj
+++ b/src/metabase/driver/generic_sql.clj
@@ -338,7 +338,9 @@
             :foreign-keys
             :expressions
             :expression-aggregations
-            :native-parameters}
+            :native-parameters
+            :nested-queries
+            :binning}
     (set-timezone-sql driver) (conj :set-timezone)))
 
 
diff --git a/src/metabase/driver/generic_sql/query_processor.clj b/src/metabase/driver/generic_sql/query_processor.clj
index ef2ec517ad29b6d893f98050fe9fe0551fcf2afd..8f1876938a95807418f2ab99feed9bd89268ccbd 100644
--- a/src/metabase/driver/generic_sql/query_processor.clj
+++ b/src/metabase/driver/generic_sql/query_processor.clj
@@ -1,6 +1,7 @@
 (ns metabase.driver.generic-sql.query-processor
   "The Query Processor is responsible for translating the Metabase Query Language into HoneySQL SQL forms."
   (:require [clojure.java.jdbc :as jdbc]
+            [clojure.string :as str]
             [clojure.tools.logging :as log]
             [honeysql
              [core :as hsql]
@@ -17,12 +18,19 @@
             [metabase.util.honeysql-extensions :as hx])
   (:import clojure.lang.Keyword
            java.sql.SQLException
-           [metabase.query_processor.interface AgFieldRef BinnedField DateTimeField DateTimeValue Expression ExpressionRef Field RelativeDateTimeValue Value]))
+           [metabase.query_processor.interface AgFieldRef BinnedField DateTimeField DateTimeValue Expression ExpressionRef Field FieldLiteral RelativeDateTimeValue Value]))
 
 (def ^:dynamic *query*
   "The outer query currently being processed."
   nil)
 
+(def ^:private ^:dynamic *nested-query-level*
+  "How many levels deep are we into nested queries? (0 = top level.)
+   We keep track of this so we know what level to find referenced aggregations
+  (otherwise something like [:aggregate-field 0] could be ambiguous in a nested query).
+  Each nested query increments this counter by 1."
+  0)
+
 (defn- driver [] {:pre [(map? *query*)]} (:driver *query*))
 
 ;; register the function "distinct-count" with HoneySQL
@@ -33,7 +41,7 @@
 
 ;;; ## Formatting
 
-(defn qualified-alias
+(defn- qualified-alias
   "Convert the given `FIELD` to a stringified alias"
   [field]
   (some->> field
@@ -54,6 +62,17 @@
   (or (get-in *query* [:query :expressions (keyword expression-name)]) (:expressions (:query *query*))
       (throw (Exception. (format "No expression named '%s'." (name expression-name))))))
 
+(defn- aggregation-at-index
+  "Fetch the aggregation at index. This is intended to power aggregate field references (e.g. [:aggregate-field 0]).
+   This also handles nested queries, which could be potentially ambiguous if multiple levels had aggregations."
+  ([index]
+   (aggregation-at-index index (:query *query*) *nested-query-level*))
+  ;; keep recursing deeper into the query until we get to the same level the aggregation reference was defined at
+  ([index query aggregation-level]
+   (if (zero? aggregation-level)
+     (nth (:aggregation query) index)
+     (recur index (:source-query query) (dec aggregation-level)))))
+
 ;; TODO - maybe this fn should be called `->honeysql` instead.
 (defprotocol ^:private IGenericSQLFormattable
   (formatted [this]
@@ -84,6 +103,10 @@
         (isa? special-type :type/UNIXTimestampMilliseconds) (sql/unix-timestamp->timestamp (driver) field :milliseconds)
         :else                                               field)))
 
+  FieldLiteral
+  (formatted [{:keys [field-name]}]
+    (keyword (hx/escape-dots (name field-name))))
+
   DateTimeField
   (formatted [{unit :unit, field :field}]
     (sql/date (driver) unit (formatted field)))
@@ -106,7 +129,7 @@
   ;; e.g. the ["aggregation" 0] fields we allow in order-by
   AgFieldRef
   (formatted [{index :index}]
-    (let [{:keys [aggregation-type]} (nth (:aggregation (:query *query*)) index)]
+    (let [{:keys [aggregation-type]} (aggregation-at-index index)]
       ;; For some arcane reason we name the results of a distinct aggregation "count",
       ;; everything else is named the same as the aggregation
       (if (= aggregation-type :distinct)
@@ -267,12 +290,24 @@
   {:pre [table-name]}
   (h/from honeysql-form (hx/qualify-and-escape-dots schema table-name)))
 
+(declare apply-clauses)
+
+(defn- apply-source-query [driver honeysql-form {{:keys [native], :as source-query} :source-query}]
+  ;; TODO - what alias should we give the source query?
+  (assoc honeysql-form
+    :from [[(if native
+              (hsql/raw (str "(" (str/replace native #";+\s*$" "") ")")) ; strip off any trailing slashes
+              (binding [*nested-query-level* (inc *nested-query-level*)]
+                (apply-clauses driver {} source-query)))
+            :source]]))
+
 (def ^:private clause-handlers
   ;; 1) Use the vars rather than the functions themselves because them implementation
   ;;    will get swapped around and  we'll be left with old version of the function that nobody implements
   ;; 2) This is a vector rather than a map because the order the clauses get handled is important for some drivers.
   ;;    For example, Oracle needs to wrap the entire query in order to apply its version of limit (`WHERE ROWNUM`).
   [:source-table (u/drop-first-arg apply-source-table)
+   :source-query apply-source-query
    :aggregation  #'sql/apply-aggregation
    :breakout     #'sql/apply-breakout
    :fields       #'sql/apply-fields
@@ -291,7 +326,8 @@
                           honeysql-form)]
       (if (seq more)
         (recur honeysql-form more)
-        honeysql-form))))
+        ;; ok, we're done; if no `:select` clause was specified (for whatever reason) put a default (`SELECT *`) one in
+        (update honeysql-form :select #(if (seq %) % [:*]))))))
 
 
 (defn build-honeysql-form
diff --git a/src/metabase/driver/googleanalytics.clj b/src/metabase/driver/googleanalytics.clj
index e2b0cb51af1dfaee89d85ee8add22c9e050de779..0fff9fe3927e35425a1e01d6f3093996aff6827b 100644
--- a/src/metabase/driver/googleanalytics.clj
+++ b/src/metabase/driver/googleanalytics.clj
@@ -247,5 +247,7 @@
           :mbql->native             (u/drop-first-arg qp/mbql->native)
           :table-rows-seq           (u/drop-first-arg table-rows-seq)}))
 
-
-(driver/register-driver! :googleanalytics (GoogleAnalyticsDriver.))
+(defn -init-driver
+  "Register the Google Analytics driver"
+  []
+  (driver/register-driver! :googleanalytics (GoogleAnalyticsDriver.)))
diff --git a/src/metabase/driver/h2.clj b/src/metabase/driver/h2.clj
index ff35bb0c5b0bc6f4e877983706d367f18cba9d45..17cd9360d55e2fe6a2e06f4e56832241f69ee11a 100644
--- a/src/metabase/driver/h2.clj
+++ b/src/metabase/driver/h2.clj
@@ -225,4 +225,7 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-(driver/register-driver! :h2 (H2Driver.))
+(defn -init-driver
+  "Register the H2 driver"
+  []
+  (driver/register-driver! :h2 (H2Driver.)))
diff --git a/src/metabase/driver/mongo.clj b/src/metabase/driver/mongo.clj
index 3b51ad2964b71c8a3603da2ae268fed2dad8fcdb..176593d4f20820a750d7d7786e142f6cefbe472f 100644
--- a/src/metabase/driver/mongo.clj
+++ b/src/metabase/driver/mongo.clj
@@ -218,5 +218,7 @@
           :process-query-in-context          (u/drop-first-arg process-query-in-context)
           :sync-in-context                   (u/drop-first-arg sync-in-context)}))
 
-
-(driver/register-driver! :mongo (MongoDriver.))
+(defn -init-driver
+  "Register the MongoDB driver"
+  []
+  (driver/register-driver! :mongo (MongoDriver.)))
diff --git a/src/metabase/driver/mysql.clj b/src/metabase/driver/mysql.clj
index 0c4c4a02b20da857011a5b95d24e5926f45a0e4e..7b2f6faad108b7cfdf5d097bad5ca4d6018288e9 100644
--- a/src/metabase/driver/mysql.clj
+++ b/src/metabase/driver/mysql.clj
@@ -198,4 +198,7 @@
           :set-timezone-sql          (constantly "SET @@session.time_zone = %s;")
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-(driver/register-driver! :mysql (MySQLDriver.))
+(defn -init-driver
+  "Register the MySQL driver"
+  []
+  (driver/register-driver! :mysql (MySQLDriver.)))
diff --git a/src/metabase/driver/oracle.clj b/src/metabase/driver/oracle.clj
index 8b1bb29f3d5a4780bbcac8cc5acd01f5602f21ee..2309c2fa228dd78ff3357f114d44d0e92ca55ff7 100644
--- a/src/metabase/driver/oracle.clj
+++ b/src/metabase/driver/oracle.clj
@@ -156,7 +156,8 @@
 (defn- apply-limit [honeysql-query {value :limit}]
   {:pre [(integer? value)]}
   {:select [:*]
-   :from   [honeysql-query]
+   :from   [(merge {:select [:*]}   ; if `honeysql-query` doesn't have a `SELECT` clause yet (which might be the case when using a source query)
+                   honeysql-query)] ; fall back to including a `SELECT *` just to make sure a valid query is produced
    :where  [:<= (hsql/raw "rownum") value]})
 
 (defn- apply-page [honeysql-query {{:keys [items page]} :page}]
@@ -167,7 +168,9 @@
       ;; if we need to do an offset we have to do double-nesting
       {:select [:*]
        :from   [{:select [:__table__.* [(hsql/raw "rownum") :__rownum__]]
-                 :from   [[honeysql-query :__table__]]
+                 :from   [[(merge {:select [:*]}
+                                  honeysql-query)
+                           :__table__]]
                  :where  [:<= (hsql/raw "rownum") (+ offset items)]}]
        :where  [:> :__rownum__ offset]})))
 
@@ -281,11 +284,14 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-;; only register the Oracle driver if the JDBC driver is available
-(when (u/ignore-exceptions
-        (Class/forName "oracle.jdbc.OracleDriver"))
-  ;; By default the Oracle JDBC driver isn't compliant with JDBC standards -- instead of returning types like java.sql.Timestamp
-  ;; it returns wacky types like oracle.sql.TIMESTAMPT. By setting this System property the JDBC driver will return the appropriate types.
-  ;; See this page for more details: http://docs.oracle.com/database/121/JJDBC/datacc.htm#sthref437
-  (.setProperty (System/getProperties) "oracle.jdbc.J2EE13Compliant" "TRUE")
-  (driver/register-driver! :oracle (OracleDriver.)))
+(defn -init-driver
+  "Register the oracle driver when the JAR is found on the classpath"
+  []
+  ;; only register the Oracle driver if the JDBC driver is available
+  (when (u/ignore-exceptions
+         (Class/forName "oracle.jdbc.OracleDriver"))
+    ;; By default the Oracle JDBC driver isn't compliant with JDBC standards -- instead of returning types like java.sql.Timestamp
+    ;; it returns wacky types like oracle.sql.TIMESTAMPT. By setting this System property the JDBC driver will return the appropriate types.
+    ;; See this page for more details: http://docs.oracle.com/database/121/JJDBC/datacc.htm#sthref437
+    (.setProperty (System/getProperties) "oracle.jdbc.J2EE13Compliant" "TRUE")
+    (driver/register-driver! :oracle (OracleDriver.))))
diff --git a/src/metabase/driver/postgres.clj b/src/metabase/driver/postgres.clj
index d823191bb9b698dc82e169358547cb95e3e2d030..b81eaf4534a3798eb89ca1cf57095645f903fce9 100644
--- a/src/metabase/driver/postgres.clj
+++ b/src/metabase/driver/postgres.clj
@@ -233,4 +233,7 @@
 
   sql/ISQLDriver PostgresISQLDriverMixin)
 
-(driver/register-driver! :postgres (PostgresDriver.))
+(defn -init-driver
+  "Register the PostgreSQL driver"
+  []
+  (driver/register-driver! :postgres (PostgresDriver.)))
diff --git a/src/metabase/driver/presto.clj b/src/metabase/driver/presto.clj
index 87efc4f5c99948683d29205e302d23ab193e1192..aa4bf5fbaa99e08683f3ae6571cf0f34efe6f9f6 100644
--- a/src/metabase/driver/presto.clj
+++ b/src/metabase/driver/presto.clj
@@ -62,21 +62,20 @@
   (let [parsers (map (comp field-type->parser :type) columns)]
     (for [row data]
       (for [[value parser] (partition 2 (interleave row parsers))]
-        (when value
+        (when (some? value)
           (parser value))))))
 
 (defn- fetch-presto-results! [details {prev-columns :columns, prev-rows :rows} uri]
-  (ssh/with-ssh-tunnel [details-with-tunnel details]
-    (let [{{:keys [columns data nextUri error]} :body} (http/get uri (assoc (details->request details-with-tunnel) :as :json))]
-      (when error
-        (throw (ex-info (or (:message error) "Error running query.") error)))
-      (let [rows    (parse-presto-results columns data)
-            results {:columns (or columns prev-columns)
-                     :rows    (vec (concat prev-rows rows))}]
-        (if (nil? nextUri)
-          results
-          (do (Thread/sleep 100) ; Might not be the best way, but the pattern is that we poll Presto at intervals
-              (fetch-presto-results! details-with-tunnel results nextUri)))))))
+  (let [{{:keys [columns data nextUri error]} :body} (http/get uri (assoc (details->request details) :as :json))]
+    (when error
+      (throw (ex-info (or (:message error) "Error running query.") error)))
+    (let [rows    (parse-presto-results columns data)
+          results {:columns (or columns prev-columns)
+                   :rows    (vec (concat prev-rows rows))}]
+      (if (nil? nextUri)
+        results
+        (do (Thread/sleep 100) ; Might not be the best way, but the pattern is that we poll Presto at intervals
+            (fetch-presto-results! details results nextUri))))))
 
 (defn- execute-presto-query! [details query]
   (ssh/with-ssh-tunnel [details-with-tunnel details]
@@ -100,6 +99,13 @@
 (defn- quote+combine-names [& names]
   (str/join \. (map quote-name names)))
 
+(defn- rename-duplicates [values]
+  ;; Appends _2, _3 and so on to duplicated values
+  (loop [acc [], [h & tail] values, seen {}]
+    (let [value (if (seen h) (str h "_" (inc (seen h))) h)]
+      (if tail
+        (recur (conj acc value) tail (assoc seen h (inc (get seen h 0))))
+        (conj acc value)))))
 
 ;;; IDriver implementation
 
@@ -192,10 +198,13 @@
 
 (defn- execute-query [{:keys [database settings], {sql :query, params :params} :native, :as outer-query}]
   (let [sql                    (str "-- " (qputil/query->remark outer-query) "\n"
-                                          (unprepare/unprepare (cons sql params) :quote-escape "'", :iso-8601-fn  :from_iso8601_timestamp))
+                                          (unprepare/unprepare (cons sql params) :quote-escape "'", :iso-8601-fn :from_iso8601_timestamp))
         details                (merge (:details database) settings)
-        {:keys [columns rows]} (execute-presto-query! details sql)]
-    {:columns (map (comp keyword :name) columns)
+        {:keys [columns rows]} (execute-presto-query! details sql)
+        columns                (for [[col name] (map vector columns (rename-duplicates (map :name columns)))]
+                                 {:name name, :base_type (presto-type->base-type (:type col))})]
+    {:cols    columns
+     :columns (map (comp keyword :name) columns)
      :rows    rows}))
 
 (defn- field-values-lazy-seq [{field-name :name, :as field}]
@@ -325,7 +334,8 @@
                                                                       :standard-deviation-aggregations
                                                                       :expressions
                                                                       :native-parameters
-                                                                      :expression-aggregations}
+                                                                      :expression-aggregations
+                                                                      :binning}
                                                                     (when-not config/is-test?
                                                                       ;; during unit tests don't treat presto as having FK support
                                                                       #{:foreign-keys})))
@@ -348,5 +358,7 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-
-(driver/register-driver! :presto (PrestoDriver.))
+(defn -init-driver
+  "Register the Presto driver"
+  []
+  (driver/register-driver! :presto (PrestoDriver.)))
diff --git a/src/metabase/driver/redshift.clj b/src/metabase/driver/redshift.clj
index b477443110670cbcb6f5d2150d33b2ccc0abaf2e..0379e2524582a853eab52bf6b1672c36d240aaa1 100644
--- a/src/metabase/driver/redshift.clj
+++ b/src/metabase/driver/redshift.clj
@@ -108,4 +108,7 @@
                                                 (str "schema_" i))
                                               "public")))))})))
 
-(driver/register-driver! :redshift (RedshiftDriver.))
+(defn -init-driver
+  "Register the Redshift driver"
+  []
+  (driver/register-driver! :redshift (RedshiftDriver.)))
diff --git a/src/metabase/driver/sqlite.clj b/src/metabase/driver/sqlite.clj
index 9b07a7c53d063ca558c395e979db989e4c970ee3..e1ef8fae02615c3e6a6a24e48c68aacab6bbc829 100644
--- a/src/metabase/driver/sqlite.clj
+++ b/src/metabase/driver/sqlite.clj
@@ -179,4 +179,7 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-(driver/register-driver! :sqlite (SQLiteDriver.))
+(defn -init-driver
+  "Register the SQLite driver"
+  []
+  (driver/register-driver! :sqlite (SQLiteDriver.)))
diff --git a/src/metabase/driver/sqlserver.clj b/src/metabase/driver/sqlserver.clj
index 1176251abf18bd1196c7463345eabaf42444906d..0190f65567e841ae30b7fab21f371fa01538b09c 100644
--- a/src/metabase/driver/sqlserver.clj
+++ b/src/metabase/driver/sqlserver.clj
@@ -192,4 +192,7 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-(driver/register-driver! :sqlserver (SQLServerDriver.))
+(defn -init-driver
+  "Register the SQLServer driver"
+  []
+  (driver/register-driver! :sqlserver (SQLServerDriver.)))
diff --git a/src/metabase/driver/vertica.clj b/src/metabase/driver/vertica.clj
index 855710c32efc68f2eef4423b81e42fe80f3430e6..3c117d27e427914350ab6994b9d907c6e50d3160 100644
--- a/src/metabase/driver/vertica.clj
+++ b/src/metabase/driver/vertica.clj
@@ -47,7 +47,17 @@
     :seconds      (hsql/call :to_timestamp expr)
     :milliseconds (recur (hx// expr 1000) :seconds)))
 
-(defn- date-trunc [unit expr] (hsql/call :date_trunc (hx/literal unit) expr))
+(defn- cast-timestamp
+  "Vertica requires stringified timestamps (what
+  Date/DateTime/Timestamps are converted to) to be cast as timestamps
+  before date operations can be performed. This function will add that
+  cast if it is a timestamp, otherwise this is a noop."
+  [expr]
+  (if (u/is-temporal? expr)
+    (hx/cast :timestamp expr)
+    expr))
+
+(defn- date-trunc [unit expr] (hsql/call :date_trunc (hx/literal unit) (cast-timestamp expr)))
 (defn- extract    [unit expr] (hsql/call :extract    unit              expr))
 
 (def ^:private extract-integer (comp hx/->integer extract))
@@ -65,7 +75,8 @@
     :day-of-week     (hx/inc (extract-integer :dow expr))
     :day-of-month    (extract-integer :day expr)
     :day-of-year     (extract-integer :doy expr)
-    :week            (hx/- (date-trunc :week (hx/+ expr one-day))
+    :week            (hx/- (date-trunc :week (hx/+ (cast-timestamp expr)
+                                                   one-day))
                            one-day)
     ;:week-of-year    (extract-integer :week (hx/+ expr one-day))
     :week-of-year    (hx/week expr)
@@ -134,8 +145,10 @@
           :string-length-fn          (u/drop-first-arg string-length-fn)
           :unix-timestamp->timestamp (u/drop-first-arg unix-timestamp->timestamp)}))
 
-
-;; only register the Vertica driver if the JDBC driver is available
-(when (u/ignore-exceptions
-        (Class/forName "com.vertica.jdbc.Driver"))
-  (driver/register-driver! :vertica (VerticaDriver.)))
+(defn -init-driver
+  "Register the Vertica driver when found on the classpath"
+  []
+  ;; only register the Vertica driver if the JDBC driver is available
+  (when (u/ignore-exceptions
+         (Class/forName "com.vertica.jdbc.Driver"))
+    (driver/register-driver! :vertica (VerticaDriver.))))
diff --git a/src/metabase/events/activity_feed.clj b/src/metabase/events/activity_feed.clj
index 878a30038731898ea14f5b62647146dbd450986a..f233c5d4d7106459637bb3bda6666d969291fb91 100644
--- a/src/metabase/events/activity_feed.clj
+++ b/src/metabase/events/activity_feed.clj
@@ -1,12 +1,16 @@
 (ns metabase.events.activity-feed
   (:require [clojure.core.async :as async]
             [clojure.tools.logging :as log]
-            [metabase.events :as events]
+            [metabase
+             [events :as events]
+             [query-processor :as qp]
+             [util :as u]]
             [metabase.models
              [activity :as activity :refer [Activity]]
              [card :refer [Card]]
              [dashboard :refer [Dashboard]]
              [table :as table]]
+            [metabase.query-processor.util :as qputil]
             [toucan.db :as db]))
 
 (def ^:const activity-feed-topics
@@ -36,11 +40,20 @@
 
 ;;; ## ---------------------------------------- EVENT PROCESSING ----------------------------------------
 
+(defn- inner-query->source-table-id
+  "Recurse through INNER-QUERY source-queries as needed until we can return the ID of this query's source-table."
+  [inner-query]
+  (or (when-let [source-table (qputil/get-normalized inner-query :source-table)]
+        (u/get-id source-table))
+      (when-let [source-query (qputil/get-normalized inner-query :source-query)]
+        (recur source-query))))
 
 (defn- process-card-activity! [topic object]
   (let [details-fn  #(select-keys % [:name :description])
-        database-id (get-in object [:dataset_query :database])
-        table-id    (get-in object [:dataset_query :query :source_table])]
+        query       (u/ignore-exceptions (qp/expand (:dataset_query object)))
+        database-id (when-let [database (:database query)]
+                      (u/get-id database))
+        table-id    (inner-query->source-table-id (:query query))]
     (activity/record-activity!
       :topic       topic
       :object      object
diff --git a/src/metabase/integrations/ldap.clj b/src/metabase/integrations/ldap.clj
new file mode 100644
index 0000000000000000000000000000000000000000..a94ff30f5f98160c122d002171fc10dd895c9639
--- /dev/null
+++ b/src/metabase/integrations/ldap.clj
@@ -0,0 +1,225 @@
+(ns metabase.integrations.ldap
+  (:require [clj-ldap.client :as ldap]
+            [clojure
+             [set :as set]
+             [string :as str]]
+            [metabase.models
+             [permissions-group :as group :refer [PermissionsGroup]]
+             [setting :as setting :refer [defsetting]]
+             [user :as user :refer [User]]]
+            [metabase.util :as u]
+            [toucan.db :as db])
+  (:import [com.unboundid.ldap.sdk LDAPConnectionPool LDAPException]))
+
+(def ^:private filter-placeholder
+  "{login}")
+
+(defsetting ldap-enabled
+  "Enable LDAP authentication."
+  :type    :boolean
+  :default false)
+
+(defsetting ldap-host
+  "Server hostname.")
+
+(defsetting ldap-port
+  "Server port, usually 389 or 636 if SSL is used."
+  :default "389")
+
+(defsetting ldap-security
+  "Use SSL, TLS or plain text."
+  :default "none"
+  :setter  (fn [new-value]
+             (when-not (nil? new-value)
+               (assert (contains? #{"none" "ssl" "starttls"} new-value)))
+             (setting/set-string! :ldap-security new-value)))
+
+(defsetting ldap-bind-dn
+  "The Distinguished Name to bind as (if any), this user will be used to lookup information about other users.")
+
+(defsetting ldap-password
+  "The password to bind with for the lookup user.")
+
+(defsetting ldap-user-base
+  "Search base for users. (Will be searched recursively)")
+
+(defsetting ldap-user-filter
+  "User lookup filter, the placeholder {login} will be replaced by the user supplied login."
+  :default "(&(objectClass=inetOrgPerson)(|(uid={login})(mail={login})))")
+
+(defsetting ldap-attribute-email
+  "Attribute to use for the user's email. (usually 'mail', 'email' or 'userPrincipalName')"
+  :default "mail")
+
+(defsetting ldap-attribute-firstname
+  "Attribute to use for the user's first name. (usually 'givenName')"
+  :default "givenName")
+
+(defsetting ldap-attribute-lastname
+  "Attribute to use for the user's last name. (usually 'sn')"
+  :default "sn")
+
+(defsetting ldap-group-sync
+  "Enable group membership synchronization with LDAP."
+  :type    :boolean
+  :default false)
+
+(defsetting ldap-group-base
+  "Search base for groups, not required if your LDAP directory provides a 'memberOf' overlay. (Will be searched recursively)")
+
+(defsetting ldap-group-mappings
+  ;; Should be in the form: {"cn=Some Group,dc=...": [1, 2, 3]} where keys are LDAP groups and values are lists of MB groups IDs
+  "JSON containing LDAP to Metabase group mappings."
+  :type    :json
+  :default {})
+
+(defn ldap-configured?
+  "Check if LDAP is enabled and that the mandatory settings are configured."
+  []
+  (boolean (and (ldap-enabled)
+                (ldap-host)
+                (ldap-user-base))))
+
+(defn- details->ldap-options [{:keys [host port bind-dn password security]}]
+  {:host      (str host ":" port)
+   :bind-dn   bind-dn
+   :password  password
+   :ssl?      (= security "ssl")
+   :startTLS? (= security "starttls")})
+
+(defn- settings->ldap-options []
+  (details->ldap-options {:host      (ldap-host)
+                          :port      (ldap-port)
+                          :bind-dn   (ldap-bind-dn)
+                          :password  (ldap-password)
+                          :security  (ldap-security)}))
+
+(defn- escape-value
+  "Escapes a value for use in an LDAP filter expression."
+  [value]
+  (str/replace value #"[\*\(\)\\\\0]" (comp (partial format "\\%02X") int first)))
+
+(defn- get-connection
+  "Connects to LDAP with the currently set settings and returns the connection."
+  ^LDAPConnectionPool []
+  (ldap/connect (settings->ldap-options)))
+
+(defn- with-connection
+  "Applies `f` with a connection and `args`"
+  [f & args]
+  (with-open [conn (get-connection)]
+    (apply f conn args)))
+
+(defn- ldap-groups->mb-group-ids
+  "Will translate a set of DNs to a set of MB group IDs using the configured mappings."
+  [ldap-groups]
+  (-> (ldap-group-mappings)
+      (select-keys (map keyword ldap-groups))
+      vals
+      flatten
+      set))
+
+(defn- get-user-groups
+  "Retrieve groups for a supplied DN."
+  ([dn]
+    (with-connection get-user-groups dn))
+  ([conn dn]
+    (when (ldap-group-base)
+      (let [results (ldap/search conn (ldap-group-base) {:scope      :sub
+                                                         :filter     (str "member=" (escape-value dn))
+                                                         :attributes [:dn :distinguishedName]})]
+        (filter some?
+          (for [result results]
+            (or (:dn result) (:distinguishedName result))))))))
+
+(def ^:private user-base-error  {:status :ERROR, :message "User search base does not exist or is unreadable"})
+(def ^:private group-base-error {:status :ERROR, :message "Group search base does not exist or is unreadable"})
+
+(defn test-ldap-connection
+  "Test the connection to an LDAP server to determine if we can find the search base.
+
+   Takes in a dictionary of properties such as:
+       {:host       \"localhost\"
+        :port       389
+        :bind-dn    \"cn=Directory Manager\"
+        :password   \"password\"
+        :security   \"none\"
+        :user-base  \"ou=Birds,dc=metabase,dc=com\"
+        :group-base \"ou=Groups,dc=metabase,dc=com\"}"
+  [{:keys [user-base group-base], :as details}]
+  (try
+    (with-open [^LDAPConnectionPool conn (ldap/connect (details->ldap-options details))]
+      (or
+       (try
+         (when-not (ldap/get conn user-base)
+           user-base-error)
+         (catch Exception e
+           user-base-error))
+       (when group-base
+         (try
+           (when-not (ldap/get conn group-base)
+             group-base-error)
+           (catch Exception e
+             group-base-error)))
+       {:status :SUCCESS}))
+    (catch LDAPException e
+      {:status :ERROR, :message (.getMessage e), :code (.getResultCode e)})
+    (catch Exception e
+      {:status :ERROR, :message (.getMessage e)})))
+
+(defn find-user
+  "Gets user information for the supplied username."
+  ([username]
+    (with-connection find-user username))
+  ([conn username]
+    (let [fname-attr (keyword (ldap-attribute-firstname))
+          lname-attr (keyword (ldap-attribute-lastname))
+          email-attr (keyword (ldap-attribute-email))]
+      (when-let [[result] (ldap/search conn (ldap-user-base) {:scope      :sub
+                                                              :filter     (str/replace (ldap-user-filter) filter-placeholder (escape-value username))
+                                                              :attributes [:dn :distinguishedName fname-attr lname-attr email-attr :memberOf]
+                                                              :size-limit 1})]
+        (let [dn    (or (:dn result) (:distinguishedName result))
+              fname (get result fname-attr)
+              lname (get result lname-attr)
+              email (get result email-attr)]
+          ;; Make sure we got everything as these are all required for new accounts
+          (when-not (or (empty? dn) (empty? fname) (empty? lname) (empty? email))
+            ;; ActiveDirectory (and others?) will supply a `memberOf` overlay attribute for groups
+            ;; Otherwise we have to make the inverse query to get them
+            (let [groups (when (ldap-group-sync)
+                           (or (:memberOf result) (get-user-groups dn) []))]
+              {:dn         dn
+               :first-name fname
+               :last-name  lname
+               :email      email
+               :groups     groups})))))))
+
+(defn verify-password
+  "Verifies if the supplied password is valid for the `user-info` (from `find-user`) or DN."
+  ([user-info password]
+    (with-connection verify-password user-info password))
+  ([conn user-info password]
+    (if (string? user-info)
+      (ldap/bind? conn user-info password)
+      (ldap/bind? conn (:dn user-info) password))))
+
+(defn fetch-or-create-user!
+  "Using the `user-info` (from `find-user`) get the corresponding Metabase user, creating it if necessary."
+  [{:keys [first-name last-name email groups]} password]
+  (let [user (or (db/select-one [User :id :last_login] :email email)
+             (user/create-new-ldap-auth-user! first-name last-name email password))]
+    (u/prog1 user
+      (when password
+        (user/set-password! (:id user) password))
+      (when (ldap-group-sync)
+        (let [special-ids #{(:id (group/admin)) (:id (group/all-users))}
+              current-ids (set (map :group_id (db/select ['PermissionsGroupMembership :group_id] :user_id (:id user))))
+              ldap-ids    (when-let [ids (seq (ldap-groups->mb-group-ids groups))]
+                            (set (map :id (db/select [PermissionsGroup :id] :id [:in ids]))))
+              to-remove   (set/difference current-ids ldap-ids special-ids)
+              to-add      (set/difference ldap-ids current-ids)]
+          (when (seq to-remove)
+            (db/delete! 'PermissionsGroupMembership :group_id [:in to-remove], :user_id (:id user)))
+          (doseq [id to-add]
+            (db/insert! 'PermissionsGroupMembership :group_id id, :user_id (:id user))))))))
diff --git a/src/metabase/middleware.clj b/src/metabase/middleware.clj
index 9ca52e30797b1ab6cade62ca6eda95ef22a6581a..5511be6662c28fbb09f209513c9ec947c27ce773 100644
--- a/src/metabase/middleware.clj
+++ b/src/metabase/middleware.clj
@@ -123,7 +123,7 @@
       response-unauthentic)))
 
 (def ^:private current-user-fields
-  (vec (concat [User :is_active :google_auth] (models/default-fields User))))
+  (vec (concat [User :is_active :google_auth :ldap_auth] (models/default-fields User))))
 
 (defn bind-current-user
   "Middleware that binds `metabase.api.common/*current-user*`, `*current-user-id*`, `*is-superuser?*`, and `*current-user-permissions-set*`.
diff --git a/src/metabase/models/card.clj b/src/metabase/models/card.clj
index 7e2e1de81a47bf7bd42c93d5ed6fcf1ce9abd60d..f276c68303fbca7250951ece0872c0f2f701b394 100644
--- a/src/metabase/models/card.clj
+++ b/src/metabase/models/card.clj
@@ -53,10 +53,12 @@
 
 (defn- query->source-and-join-tables
   "Return a sequence of all Tables (as TableInstance maps) referenced by QUERY."
-  [{:keys [source-table join-tables native], :as query}]
+  [{:keys [source-table join-tables source-query native], :as query}]
   (cond
     ;; if we come across a native query just put a placeholder (`::native`) there so we know we need to add native permissions to the complete set below.
     native       [::native]
+    ;; if we have a source-query just recur until we hit either the native source or the MBQL source
+    source-query (recur source-query)
     ;; for root MBQL queries just return source-table + join-tables
     :else        (cons source-table join-tables)))
 
@@ -86,6 +88,7 @@
 
 ;; it takes a lot of DB calls and function calls to expand/resolve a query, and since they're pure functions we can save ourselves some a lot of DB calls
 ;; by caching the results. Cache the permissions reqquired to run a given query dictionary for up to 6 hours
+;; TODO - what if the query uses a source query, and that query changes? Not sure if that will cause an issue or not. May need to revisit this
 (defn- query-perms-set* [{query-type :type, database :database, :as query} read-or-write]
   (cond
     (= query {})                     #{}
@@ -135,12 +138,15 @@
 ;;; ------------------------------------------------------------ Lifecycle ------------------------------------------------------------
 
 (defn- query->database-and-table-ids
-  "Return a map with `:database-id` and source `:table-id` that should be saved for a Card."
+  "Return a map with `:database-id` and source `:table-id` that should be saved for a Card. Handles queries that use other queries as their source
+   (ones that come in with a `:source-table` like `card__100`) recursively, as well as normal queries."
   [outer-query]
-  (let [database     (qputil/get-normalized outer-query :database)
+  (let [database-id  (qputil/get-normalized outer-query :database)
         source-table (qputil/get-in-normalized outer-query [:query :source-table])]
-    (when source-table
-      {:database-id (u/get-id database), :table-id (u/get-id source-table)})))
+    (cond
+      (integer? source-table) {:database-id database-id, :table-id source-table}
+      (string? source-table)  (let [[_ card-id] (re-find #"^card__(\d+)$" source-table)]
+                                (db/select-one [Card [:table_id :table-id] [:database_id :database-id]] :id (Integer/parseInt card-id))))))
 
 (defn- populate-query-fields [{{query-type :type, :as outer-query} :dataset_query, :as card}]
   (merge (when query-type
@@ -184,6 +190,7 @@
                                        :display                :keyword
                                        :embedding_params       :json
                                        :query_type             :keyword
+                                       :result_metadata        :json
                                        :visualization_settings :json})
           :properties     (constantly {:timestamped? true})
           :pre-update     (comp populate-query-fields pre-update)
diff --git a/src/metabase/models/database.clj b/src/metabase/models/database.clj
index f84d61e3a169d6ebfdd73290ecd40117a694b5db..18c311e28db5b19c9b08992ba5b0661746ef4ef1 100644
--- a/src/metabase/models/database.clj
+++ b/src/metabase/models/database.clj
@@ -12,6 +12,24 @@
              [db :as db]
              [models :as models]]))
 
+
+;;; ------------------------------------------------------------ Constants ------------------------------------------------------------
+
+;; TODO - should this be renamed `saved-cards-virtual-id`?
+(def ^:const ^Integer virtual-id
+  "The ID used to signify that a database is 'virtual' rather than physical.
+
+   A fake integer ID is used so as to minimize the number of changes that need to be made on the frontend -- by using something that would otherwise
+   be a legal ID, *nothing* need change there, and the frontend can query against this 'database' none the wiser. (This integer ID is negative
+   which means it will never conflict with a *real* database ID.)
+
+   This ID acts as a sort of flag. The relevant places in the middleware can check whether the DB we're querying is this 'virtual' database and
+   take the appropriate actions."
+  -1337)
+;; To the reader: yes, this seems sort of hacky, but one of the goals of the Nested Query Initiativeâ„¢ was to minimize if not completely eliminate
+;; any changes to the frontend. After experimenting with several possible ways to do this this implementation seemed simplest and best met the goal.
+;; Luckily this is the only place this "magic number" is defined and the entire frontend can remain blissfully unaware of its value.
+
 ;;; ------------------------------------------------------------ Entity & Lifecycle ------------------------------------------------------------
 
 (models/defmodel Database :metabase_database)
@@ -60,7 +78,7 @@
 (defn ^:hydrate tables
   "Return the `Tables` associated with this `Database`."
   [{:keys [id]}]
-  (db/select 'Table, :db_id id, :active true, {:order-by [[:%lower.display_name :asc]]}))
+  (db/select 'Table, :db_id id, :active true, {:order-by [[:%lower.display_name :asc]]})) ; TODO - do we want to include tables that should be `:hidden`?
 
 (defn schema-names
   "Return a *sorted set* of schema names (as strings) associated with this `Database`."
diff --git a/src/metabase/models/field.clj b/src/metabase/models/field.clj
index 1ff8fee88f1049ed8351cc402ec8734e2a0dec79..b12e97c9e482cb32cfd9ede84a12d1d6ae767a38 100644
--- a/src/metabase/models/field.clj
+++ b/src/metabase/models/field.clj
@@ -70,7 +70,12 @@
           :properties     (constantly {:timestamped? true})
           :pre-insert     pre-insert
           :pre-update     pre-update
-          :pre-delete     pre-delete})
+          :pre-delete     pre-delete
+          :post-select    (fn [{:keys [min_value max_value] :as row}]
+                            (let [round-value #(u/round-to-decimals 4 %)]
+                              (cond-> row
+                                min_value (update :min_value round-value)
+                                max_value (update :max_value round-value))))})
   i/IObjectPermissions
   (merge i/IObjectPermissionsDefaults
          {:perms-objects-set  perms-objects-set
diff --git a/src/metabase/models/permissions.clj b/src/metabase/models/permissions.clj
index 1f1523aa0bf9dc6f28274e654b8ed714841d2238..77088bb48d44978208c675f0ebcc8fe2b7d87328 100644
--- a/src/metabase/models/permissions.clj
+++ b/src/metabase/models/permissions.clj
@@ -339,7 +339,6 @@
    (grant-permissions! group-or-id (apply object-path db-id schema more)))
   ([group-or-id path]
    (try
-     (log/debug (u/format-color 'green "Granting permissions for group %d: %s" (u/get-id group-or-id) path))
      (db/insert! Permissions
        :group_id (u/get-id group-or-id)
        :object   path)
diff --git a/src/metabase/models/permissions_group.clj b/src/metabase/models/permissions_group.clj
index b3e04e75d68eb6f46c77ad0c5d3cb370182ebf11..a146b5a95596b370c8f42447ca4d4cc6a531e2b2 100644
--- a/src/metabase/models/permissions_group.clj
+++ b/src/metabase/models/permissions_group.clj
@@ -1,6 +1,7 @@
 (ns metabase.models.permissions-group
   (:require [clojure.string :as s]
             [clojure.tools.logging :as log]
+            [metabase.models.setting :as setting]
             [metabase.util :as u]
             [toucan
              [db :as db]
@@ -69,8 +70,14 @@
 
 (defn- pre-delete [{id :id, :as group}]
   (check-not-magic-group group)
-  (db/delete! 'Permissions                :group_id id)
-  (db/delete! 'PermissionsGroupMembership :group_id id))
+  (db/delete! 'Permissions                 :group_id id)
+  (db/delete! 'PermissionsGroupMembership  :group_id id)
+  ;; Remove from LDAP mappings
+  (setting/set-json! :ldap-group-mappings
+    (when-let [mappings (setting/get-json :ldap-group-mappings)]
+      (zipmap (keys mappings)
+              (for [val (vals mappings)]
+                (remove (partial = id) val))))))
 
 (defn- pre-update [{group-name :name, :as group}]
   (u/prog1 group
diff --git a/src/metabase/models/pulse.clj b/src/metabase/models/pulse.clj
index f9517962067f1ed83540e0faa197d12277a2f8ab..d4584e058c866c2d18aaf6359ecc8771b36159d1 100644
--- a/src/metabase/models/pulse.clj
+++ b/src/metabase/models/pulse.clj
@@ -77,6 +77,7 @@
   "Return the `Cards` associated with this PULSE."
   [{:keys [id]}]
   (db/select [Card :id :name :description :display]
+    :archived false
     (mdb/join [Card :id] [PulseCard :card_id])
     (db/qualify PulseCard :pulse_id) id
     {:order-by [[(db/qualify PulseCard :position) :asc]]}))
@@ -119,7 +120,7 @@
   (db/delete! PulseCard :pulse_id id)
   ;; now just insert all of the cards that were given to us
   (when (seq card-ids)
-    (let [cards (map-indexed (fn [idx itm] {:pulse_id id :card_id itm :position idx}) card-ids)]
+    (let [cards (map-indexed (fn [i card-id] {:pulse_id id, :card_id card-id, :position i}) card-ids)]
       (db/insert-many! PulseCard cards))))
 
 
@@ -207,7 +208,7 @@
   (db/transaction
     ;; update the pulse itself
     (db/update! Pulse id, :name name, :skip_if_empty skip-if-empty?)
-    ;; update cards (only if they changed)
+    ;; update cards (only if they changed). Order for the cards is important which is why we're not using select-field
     (when (not= cards (map :card_id (db/select [PulseCard :card_id], :pulse_id id, {:order-by [[:position :asc]]})))
       (update-pulse-cards! pulse cards))
     ;; update channels
diff --git a/src/metabase/models/raw_column.clj b/src/metabase/models/raw_column.clj
index 7a9e50eaf190e6af9528d51318462d50659eb757..ae91c006a15196df7762418de3fa4372cd6ada73 100644
--- a/src/metabase/models/raw_column.clj
+++ b/src/metabase/models/raw_column.clj
@@ -1,10 +1,10 @@
-(ns metabase.models.raw-column
+(ns ^:deprecated metabase.models.raw-column
   (:require [metabase.util :as u]
             [toucan
              [db :as db]
              [models :as models]]))
 
-(models/defmodel RawColumn :raw_column)
+(models/defmodel ^:deprecated RawColumn :raw_column)
 
 (defn- pre-insert [table]
   (let [defaults {:active  true
diff --git a/src/metabase/models/raw_table.clj b/src/metabase/models/raw_table.clj
index b462de633dd62910ac88b993ebcd5074ca92d997..880d5a90cfdff999e2127540846837127653fdea 100644
--- a/src/metabase/models/raw_table.clj
+++ b/src/metabase/models/raw_table.clj
@@ -1,11 +1,11 @@
-(ns metabase.models.raw-table
+(ns ^:deprecated metabase.models.raw-table
   (:require [metabase.models.raw-column :refer [RawColumn]]
             [metabase.util :as u]
             [toucan
              [db :as db]
              [models :as models]]))
 
-(models/defmodel RawTable :raw_table)
+(models/defmodel ^:deprecated RawTable :raw_table)
 
 (defn- pre-insert [table]
   (let [defaults {:details {}}]
diff --git a/src/metabase/models/table.clj b/src/metabase/models/table.clj
index afebba6ca0ff40ed7c1d8473582d03fc1b3e9aa0..c72a0eb0f45245c94336677f1c8b18771972a8a3 100644
--- a/src/metabase/models/table.clj
+++ b/src/metabase/models/table.clj
@@ -23,7 +23,8 @@
   #{:person :event :photo :place})
 
 (def ^:const visibility-types
-  "Valid values for `Table.visibility_type` (field may also be `nil`)."
+  "Valid values for `Table.visibility_type` (field may also be `nil`).
+   (Basically any non-nil value is a reason for hiding the table.)"
   #{:hidden :technical :cruft})
 
 
diff --git a/src/metabase/models/user.clj b/src/metabase/models/user.clj
index 6e147cda3c8ff6fdf27e327e2ffe2afe81c3ac74..488a39e7a29a85a2361c3ba499a510b8ad139b43 100644
--- a/src/metabase/models/user.clj
+++ b/src/metabase/models/user.clj
@@ -143,7 +143,15 @@
     ;; send an email to everyone including the site admin if that's set
     (email/send-user-joined-admin-notification-email! <>, :google-auth? true)))
 
-
+(defn create-new-ldap-auth-user!
+  "Convenience for creating a new user via LDAP. This account is considered active immediately; thus all active admins will recieve an email right away."
+  [first-name last-name email-address password]
+  {:pre [(string? first-name) (string? last-name) (u/is-email? email-address)]}
+  (db/insert! User :email      email-address
+                   :first_name first-name
+                   :last_name  last-name
+                   :password   password
+                   :ldap_auth  true))
 
 (defn set-password!
   "Updates the stored password for a specified `User` by hashing the password with a random salt."
diff --git a/src/metabase/public_settings.clj b/src/metabase/public_settings.clj
index 53896a9fc03e2447f8bbed36a445362583695e23..545ede867435bcb59d6a5e27e6d890faaad22c5b 100644
--- a/src/metabase/public_settings.clj
+++ b/src/metabase/public_settings.clj
@@ -133,6 +133,7 @@
    :engines               ((resolve 'metabase.driver/available-drivers))
    :ga_code               "UA-60817802-1"
    :google_auth_client_id (setting/get :google-auth-client-id)
+   :ldap_configured       ((resolve 'metabase.integrations.ldap/ldap-configured?))
    :has_sample_dataset    (db/exists? 'Database, :is_sample true)
    :map_tile_server_url   (map-tile-server-url)
    :password_complexity   password/active-password-complexity
@@ -141,6 +142,7 @@
    :report_timezone       (setting/get :report-timezone)
    :setup_token           ((resolve 'metabase.setup/token-value))
    :site_name             (site-name)
+   :site_url              (site-url)
    :timezone_short        (short-timezone-name (setting/get :report-timezone))
    :timezones             common/timezones
    :types                 (types/types->parents)
diff --git a/src/metabase/pulse.clj b/src/metabase/pulse.clj
index f4168a9998800eeb04e75d1d962fa0ffb5d07bc2..82beac6a13ab23bc0cfbc05509f4fcb44bb87d65 100644
--- a/src/metabase/pulse.clj
+++ b/src/metabase/pulse.clj
@@ -19,7 +19,7 @@
   "Execute the query for a single card with CARD-ID. OPTIONS are passed along to `dataset-query`."
   [card-id & {:as options}]
   {:pre [(integer? card-id)]}
-  (when-let [card (Card card-id)]
+  (when-let [card (Card :id card-id, :archived false)]
     (let [{:keys [creator_id dataset_query]} card]
       (try
         {:card   card
@@ -86,8 +86,10 @@
        (send-pulse! pulse :channel-ids [312])    Send only to Channel with :id = 312"
   [{:keys [cards], :as pulse} & {:keys [channel-ids]}]
   {:pre [(map? pulse) (every? map? cards) (every? :id cards)]}
-  (let [results     (for [card cards]
-                      (execute-card (:id card), :pulse-id (:id pulse))) ; Pulse ID may be `nil` if the Pulse isn't saved yet
+  (let [results     (for [card  cards
+                          :let  [result (execute-card (:id card), :pulse-id (:id pulse))] ; Pulse ID may be `nil` if the Pulse isn't saved yet
+                          :when result]                                                   ; some cards may return empty results, e.g. if the card has been archived
+                      result)
         channel-ids (or channel-ids (mapv :id (:channels pulse)))]
     (when-not (and (:skip_if_empty pulse) (are-all-cards-empty? results))
       (doseq [channel-id channel-ids]
diff --git a/src/metabase/query_processor.clj b/src/metabase/query_processor.clj
index 1594d6b26f240a30310e1ef63fcd0d6176367dee..702974ddd4453de105868c80c84e0b35195dd1ed 100644
--- a/src/metabase/query_processor.clj
+++ b/src/metabase/query_processor.clj
@@ -20,12 +20,14 @@
              [driver-specific :as driver-specific]
              [expand-macros :as expand-macros]
              [expand-resolve :as expand-resolve]
+             [fetch-source-query :as fetch-source-query]
              [format-rows :as format-rows]
              [limit :as limit]
              [log :as log-query]
              [mbql-to-native :as mbql-to-native]
              [parameters :as parameters]
              [permissions :as perms]
+             [results-metadata :as results-metadata]
              [resolve-driver :as resolve-driver]]
             [metabase.query-processor.util :as qputil]
             [metabase.util.schema :as su]
@@ -87,6 +89,7 @@
       implicit-clauses/add-implicit-clauses
       format-rows/format-rows
       binning/update-binning-strategy
+      results-metadata/record-and-return-metadata!
       expand-resolve/expand-resolve                    ; ▲▲▲ QUERY EXPANSION POINT  ▲▲▲ All functions *above* will see EXPANDED query during PRE-PROCESSING
       row-count-and-status/add-row-count-and-status    ; ▼▼▼ RESULTS WRAPPING POINT ▼▼▼ All functions *below* will see results WRAPPED in `:data` during POST-PROCESSING
       parameters/substitute-parameters
@@ -94,8 +97,10 @@
       driver-specific/process-query-in-context         ; (drivers can inject custom middleware if they implement IDriver's `process-query-in-context`)
       add-settings/add-settings
       resolve-driver/resolve-driver                    ; ▲▲▲ DRIVER RESOLUTION POINT ▲▲▲ All functions *above* will have access to the driver during PRE- *and* POST-PROCESSING
+      fetch-source-query/fetch-source-query
       log-query/log-initial-query
       cache/maybe-return-cached-results
+      log-query/log-results-metadata
       catch-exceptions/catch-exceptions))
 ;; ▲▲▲ PRE-PROCESSING ▲▲▲ happens from BOTTOM-TO-TOP, e.g. the results of `expand-macros` are (eventually) passed to `expand-resolve`
 
@@ -103,8 +108,10 @@
   "Return the native form for QUERY (e.g. for a MBQL query on Postgres this would return a map containing the compiled SQL form)."
   {:style/indent 0}
   [query]
-  (-> ((qp-pipeline identity) query)
-      (get-in [:data :native_form])))
+  (let [results ((qp-pipeline identity) query)]
+    (or (get-in results [:data :native_form])
+        (throw (ex-info "No native form returned."
+                 results)))))
 
 (defn process-query
   "A pipeline of various QP functions (including middleware) that are used to process MB queries."
@@ -118,7 +125,8 @@
   (->> identity
        expand-resolve/expand-resolve
        parameters/substitute-parameters
-       expand-macros/expand-macros))
+       expand-macros/expand-macros
+       fetch-source-query/fetch-source-query))
 ;; ▲▲▲ This only does PRE-PROCESSING, so it happens from bottom to top, eventually returning the preprocessed query instead of running it
 
 
@@ -232,7 +240,8 @@
                   (s/optional-key :executed-by)  (s/maybe su/IntGreaterThanZero)
                   (s/optional-key :card-id)      (s/maybe su/IntGreaterThanZero)
                   (s/optional-key :dashboard-id) (s/maybe su/IntGreaterThanZero)
-                  (s/optional-key :pulse-id)     (s/maybe su/IntGreaterThanZero)}
+                  (s/optional-key :pulse-id)     (s/maybe su/IntGreaterThanZero)
+                  (s/optional-key :nested?)      (s/maybe s/Bool)}
                  (fn [{:keys [executed-by]}]
                    (or (integer? executed-by)
                        *allow-queries-with-no-executor-id*))
diff --git a/src/metabase/query_processor/annotate.clj b/src/metabase/query_processor/annotate.clj
index 1deb8eebcd8009d38f3d6cd0e3c15200a1dcf932..b1915874e620cc1335d84c83ae2af3f3c70db482 100644
--- a/src/metabase/query_processor/annotate.clj
+++ b/src/metabase/query_processor/annotate.clj
@@ -21,6 +21,13 @@
 
 ;;; ## Field Resolution
 
+(defn- valid-collected-field? [keep-date-time-fields? f]
+  (or (instance? metabase.query_processor.interface.Field f)
+      (instance? metabase.query_processor.interface.FieldLiteral f)
+      (instance? metabase.query_processor.interface.ExpressionRef f)
+      (when keep-date-time-fields?
+        (instance? metabase.query_processor.interface.DateTimeField f))))
+
 (defn collect-fields
   "Return a sequence of all the `Fields` inside THIS, recursing as needed for collections.
    For maps, add or `conj` to property `:path`, recording the keypath used to reach each `Field.`
@@ -28,12 +35,9 @@
      (collect-fields {:name \"id\", ...})     -> [{:name \"id\", ...}]
      (collect-fields [{:name \"id\", ...}])   -> [{:name \"id\", ...}]
      (collect-fields {:a {:name \"id\", ...}) -> [{:name \"id\", :path [:a], ...}]"
+  {:style/indent 0}
   [this & [keep-date-time-fields?]]
-  {:post [(every? (fn [f]
-                    (or (instance? metabase.query_processor.interface.Field f)
-                        (instance? metabase.query_processor.interface.ExpressionRef f)
-                        (when keep-date-time-fields?
-                          (instance? metabase.query_processor.interface.DateTimeField f)))) %)]}
+  {:post [(every? (partial valid-collected-field? keep-date-time-fields?) %)]}
   (condp instance? this
     ;; For a DateTimeField we'll flatten it back into regular Field but include the :unit info for the frontend.
     ;; Recurse so it is otherwise handled normally
@@ -58,14 +62,37 @@
       [this parent]
       [this])
 
+    metabase.query_processor.interface.FieldLiteral
+    [(assoc this
+       :field-id           [:field-literal (:field-name this) (:base-type this)]
+       :field-display-name (humanization/name->human-readable-name (:field-name this)))]
+
     metabase.query_processor.interface.ExpressionRef
     [(assoc this :field-display-name (:expression-name this))]
 
+    ;; for every value in a map in the query we'll descend into the map and find all the fields contained therein and mark the key as each field's source.
+    ;; e.g. if we descend into the `:breakout` columns for a query each field returned will get a `:source` of `:breakout`
+    ;; The source is important since it is used to determine sort order for for columns
     clojure.lang.IPersistentMap
     (for [[k v] (seq this)
           field (collect-fields v keep-date-time-fields?)
           :when field]
-      (assoc field :source k))
+      (if (= k :source-query)
+        ;; For columns collected from a source query...
+        ;; 1) Make sure they didn't accidentally pick up an integer ID if the fields clause was added implicitly. If it does
+        ;;    the frontend won't know how to use the field since it won't match up with the same field in the "virtual" table metadata.
+        ;; 2) Keep the original `:source` rather than replacing it with `:source-query` since the frontend doesn't know what to do with that.
+        (if (= (:unit field) :year)
+          ;; if the field is broken out by year we don't want to advertise it as type/DateTime because you can't do a datetime breakout on the years that come back
+          ;; (they come back as text). So instead just tell people it's a Text column
+          (assoc field
+            :field-id [:field-literal (:field-name field) :type/Text]
+            :base-type :type/Text
+            :unit      nil)
+          (assoc field
+            :field-id [:field-literal (:field-name field) (:base-type field)]))
+        ;; For all other fields just add `:source` as described above
+        (assoc field :source k)))
 
     clojure.lang.Sequential
     (for [[i field] (m/indexed (mapcat (u/rpartial collect-fields keep-date-time-fields?) this))]
@@ -157,6 +184,18 @@
    :field-name         k
    :field-display-name (humanization/name->human-readable-name (name k))})
 
+;; TODO - I'm not 100% sure the code reaches this point any more since the `collect-fields` logic now handles nested queries
+;; maybe this is used for queries where the source query is native?
+(defn- info-for-column-from-source-query
+  "Return information about a column that comes back when we're using a source query.
+   (This is basically the same as the generic information, but we also add `:id` and `:source`
+   columns so drill-through operations can be done on it)."
+  [k & [initial-values]]
+  (let [col (generic-info-for-missing-key k initial-values)]
+    (assoc col
+      :id     [:field-literal k (:base-type col)]
+      :source :fields)))
+
 
 (defn- info-for-duplicate-field
   "The Clojure JDBC driver automatically appends suffixes like `count_2` to duplicate columns if multiple columns come back with the same name;
@@ -171,25 +210,27 @@
           fields)))
 
 (defn- info-for-missing-key
-  "Metadata for a field named K, which we weren't able to resolve normally.
-   If possible, we work around This defaults to generic information "
-  [fields k initial-values]
-  (or (info-for-duplicate-field fields k)
+  "Metadata for a field named K, which we weren't able to resolve normally."
+  [inner-query fields k initial-values]
+  (or (when (:source-query inner-query)
+        (info-for-column-from-source-query k initial-values))
+      (info-for-duplicate-field fields k)
       (generic-info-for-missing-key k initial-values)))
 
 (defn- add-unknown-fields-if-needed
   "When create info maps for any fields we didn't expect to come back from the query.
    Ideally, this should never happen, but on the off chance it does we still want to return it in the results."
-  [actual-keys initial-rows fields]
-  {:pre [(set? actual-keys) (every? keyword? actual-keys)]}
+  [inner-query actual-keys initial-rows fields]
+  {:pre [(sequential? actual-keys) (every? keyword? actual-keys)]}
   (let [expected-keys (u/prog1 (set (map :field-name fields))
                         (assert (every? keyword? <>)))
-        missing-keys  (set/difference actual-keys expected-keys)]
+        missing-keys  (set/difference (set actual-keys) expected-keys)]
     (when (seq missing-keys)
-      (log/warn (u/format-color 'yellow "There are fields we weren't expecting in the results: %s\nExpected: %s\nActual: %s"
-                  missing-keys expected-keys actual-keys)))
-    (concat fields (for [k missing-keys]
-                     (info-for-missing-key fields k (map k initial-rows))))))
+      (log/warn (u/format-color 'yellow "There are fields we (maybe) weren't expecting in the results: %s\nExpected: %s\nActual: %s"
+                  missing-keys expected-keys (set actual-keys))))
+    (concat fields (for [k     actual-keys
+                         :when (contains? missing-keys k)]
+                     (info-for-missing-key inner-query fields k (map k initial-rows))))))
 
 (defn- convert-field-to-expected-format
   "Rename keys, provide default values, etc. for FIELD so it is in the format expected by the frontend."
@@ -200,7 +241,7 @@
                   :id          nil
                   :table_id    nil}]
     (-> (merge defaults field)
-        (update :field-display-name name)
+        (update :field-display-name #(when % (name %)))
         (set/rename-keys {:base-type          :base_type
                           :field-display-name :display_name
                           :field-id           :id
@@ -219,7 +260,8 @@
   "Fetch fk info and return a function that returns the destination Field of a given Field."
   ([fields]
    (or (fk-field->dest-fn fields (for [{:keys [special_type id]} fields
-                                       :when (isa? special_type :type/FK)]
+                                       :when  (and (isa? special_type :type/FK)
+                                                   (integer? id))]
                                    id))
        (constantly nil)))
   ;; Fetch the foreign key fields whose origin is in the returned Fields, create a map of origin-field-id->destination-field-id
@@ -252,20 +294,29 @@
                         {}))))))
 
 (defn- resolve-sort-and-format-columns
-  "Collect the Fields referenced in QUERY, sort them according to the rules at the top
+  "Collect the Fields referenced in INNER-QUERY, sort them according to the rules at the top
    of this page, format them as expected by the frontend, and return the results."
-  [query result-keys initial-rows]
-  {:pre [(set? result-keys)]}
+  [inner-query result-keys initial-rows]
+  {:pre [(sequential? result-keys)]}
   (when (seq result-keys)
-    (->> (collect-fields (dissoc query :expressions))
+    (->> (collect-fields (dissoc inner-query :expressions))
+         ;; qualify the field name to make sure it matches what will come back. (For Mongo nested queries only)
          (map qualify-field-name)
-         (add-aggregate-fields-if-needed query)
+         ;; add entries for aggregate fields
+         (add-aggregate-fields-if-needed inner-query)
+         ;; make field-name a keyword
          (map (u/rpartial update :field-name keyword))
-         (add-unknown-fields-if-needed result-keys initial-rows)
-         (sort/sort-fields query)
+         ;; add entries for fields we weren't expecting
+         (add-unknown-fields-if-needed inner-query result-keys initial-rows)
+         ;; remove expected fields not present in the results, and make sure they're unique
+         (filter (comp (partial contains? (set result-keys)) :field-name))
+         ;; now sort the fields
+         (sort/sort-fields inner-query)
+         ;; remove any duplicate entires
+         (m/distinct-by :field-name)
+         ;; convert them to the format expected by the frontend
          (map convert-field-to-expected-format)
-         (filter (comp (partial contains? result-keys) :name))
-         (m/distinct-by :name)
+         ;; add FK info
          add-extra-info-to-fk-fields)))
 
 (defn annotate-and-sort
@@ -277,7 +328,7 @@
   [query {:keys [columns rows], :as results}]
   (let [row-maps (for [row rows]
                    (zipmap columns row))
-        cols    (resolve-sort-and-format-columns (:query query) (set columns) (take 10 row-maps))
+        cols    (resolve-sort-and-format-columns (:query query) (distinct columns) (take 10 row-maps))
         columns (mapv :name cols)]
     (assoc results
       :cols    (vec (for [col cols]
diff --git a/src/metabase/query_processor/expand.clj b/src/metabase/query_processor/expand.clj
index a5d159e9ba78741f1f9a2ef08e27d78e8ae9d5e4..d4c994ced272eae3874844adc7c7a616eb63d011 100644
--- a/src/metabase/query_processor/expand.clj
+++ b/src/metabase/query_processor/expand.clj
@@ -10,8 +10,8 @@
             [metabase.util :as u]
             [metabase.util.schema :as su]
             [schema.core :as s])
-  (:import [metabase.query_processor.interface AgFieldRef BetweenFilter ComparisonFilter CompoundFilter Expression ExpressionRef
-            FieldPlaceholder RelativeDatetime StringFilter Value ValuePlaceholder]))
+  (:import [metabase.query_processor.interface AgFieldRef BetweenFilter ComparisonFilter CompoundFilter DateTimeValue DateTimeField Expression
+            ExpressionRef FieldLiteral FieldPlaceholder RelativeDatetime RelativeDateTimeValue StringFilter Value ValuePlaceholder]))
 
 ;;; # ------------------------------------------------------------ Clause Handlers ------------------------------------------------------------
 
@@ -24,10 +24,17 @@
   [index :- s/Int]
   (i/map->AgFieldRef {:index index}))
 
-(s/defn ^:ql ^:always-validate field-id :- FieldPlaceholder
+(s/defn ^:ql ^:always-validate field-id :- i/AnyField
   "Create a generic reference to a `Field` with ID."
-  [id :- su/IntGreaterThanZero]
-  (i/map->FieldPlaceholder {:field-id id}))
+  [id]
+  ;; If for some reason we were passed a field literal (e.g. [field-id [field-literal ...]])
+  ;; we should technically barf but since we know what people meant we'll be nice for once and fix it for them :D
+  (if (instance? FieldLiteral id)
+    (do
+      (log/warn (u/format-color 'yellow (str "It doesn't make sense to use `field-literal` forms inside `field-id` forms.\n"
+                                             "Instead of [field-id [field-literal ...]], just do [field-literal ...].")))
+      id)
+    (i/map->FieldPlaceholder {:field-id id})))
 
 (s/defn ^:private ^:always-validate field :- i/AnyField
   "Generic reference to a `Field`. F can be an integer Field ID, or various other forms like `fk->` or `aggregation`."
@@ -37,6 +44,12 @@
         (field-id f))
     f))
 
+(s/defn ^:ql ^:always-validate field-literal :- FieldLiteral
+  "Generic reference to a Field by FIELD-NAME. This is intended for use when using nested queries so as to allow one to refer to the fields coming back from
+   the source query."
+  [field-name :- su/KeywordOrString, field-type :- su/KeywordOrString]
+  (i/map->FieldLiteral {:field-name (u/keyword->qualified-name field-name), :base-type (keyword field-type)}))
+
 (s/defn ^:ql ^:always-validate named :- i/Aggregation
   "Specify a CUSTOM-NAME to use for a top-level AGGREGATION-OR-EXPRESSION in the results.
    (This will probably be extended to support Fields in the future, but for now, only the `:aggregation` clause is supported.)"
@@ -44,12 +57,19 @@
   [aggregation-or-expression :- i/Aggregation, custom-name :- su/NonBlankString]
   (assoc aggregation-or-expression :custom-name custom-name))
 
-(s/defn ^:ql ^:always-validate datetime-field :- FieldPlaceholder
+(s/defn ^:ql ^:always-validate datetime-field :- i/AnyField
   "Reference to a `DateTimeField`. This is just a `Field` reference with an associated datetime UNIT."
-  ([f _ unit] (log/warn (u/format-color 'yellow (str "The syntax for datetime-field has changed in MBQL '98. [:datetime-field <field> :as <unit>] is deprecated. "
-                                                     "Prefer [:datetime-field <field> <unit>] instead.")))
-              (datetime-field f unit))
-  ([f unit]   (assoc (field f) :datetime-unit (qputil/normalize-token unit))))
+  ([f _ unit]
+   (log/warn (u/format-color 'yellow (str "The syntax for datetime-field has changed in MBQL '98. [:datetime-field <field> :as <unit>] is deprecated. "
+                                          "Prefer [:datetime-field <field> <unit>] instead.")))
+   (datetime-field f unit))
+  ([f unit]
+   (cond
+     (instance? DateTimeField f) f
+     (instance? FieldLiteral f)  (i/map->DateTimeField {:field f, :unit (qputil/normalize-token unit)})
+     ;; if it already has a datetime unit don't replace it with a new one (?)
+     ;; (:datetime-unit f)          f
+     :else                       (assoc (field f) :datetime-unit (qputil/normalize-token unit)))))
 
 (s/defn ^:ql ^:always-validate fk-> :- FieldPlaceholder
   "Reference to a `Field` that belongs to another `Table`. DEST-FIELD-ID is the ID of this Field, and FK-FIELD-ID is the ID of the foreign key field
@@ -62,18 +82,34 @@
   (i/assert-driver-supports :foreign-keys)
   (i/map->FieldPlaceholder {:fk-field-id fk-field-id, :field-id dest-field-id}))
 
+(defn- datetime-unit
+  "Determine the appropriate datetime unit that should be used for a field F and a value V.
+   (Sometimes the value may already have a 'default' value that should be replaced with the
+   value from the field it is being used with, e.g. in a filter clause.
+   For example when filtering by minute it is important both F and V are bucketed as minutes,
+   and thus both most have the same unit."
+  [f v]
+  (qputil/normalize-token (core/or (:datetime-unit f)
+                                   (:unit f)
+                                   (:unit v))))
 
-(s/defn ^:private ^:always-validate value :- (s/cond-pre Value ValuePlaceholder)
+(s/defn ^:private ^:always-validate value :- i/AnyValue
   "Literal value. F is the `Field` it relates to, and V is `nil`, or a boolean, string, numerical, or datetime value."
   [f v]
   (cond
-    (instance? ValuePlaceholder v) v
-    (instance? Value v)            v
-    :else                          (i/map->ValuePlaceholder {:field-placeholder (field f), :value v})))
+    (instance? ValuePlaceholder v)      v
+    (instance? Value v)                 v
+    (instance? RelativeDateTimeValue v) v
+    (instance? DateTimeValue v)         v
+    (instance? RelativeDatetime v)      (i/map->RelativeDateTimeValue (assoc v :unit (datetime-unit f v), :field (datetime-field f (datetime-unit f v))))
+    (instance? DateTimeField f)         (i/map->DateTimeValue {:value (u/->Timestamp v), :field f})
+    (instance? FieldLiteral f)          (i/map->Value {:value v, :field f})
+    :else                               (i/map->ValuePlaceholder {:field-placeholder (field f), :value v})))
 
 (s/defn ^:private ^:always-validate field-or-value
   "Use instead of `value` when something may be either a field or a value."
   [f v]
+
   (if (core/or (instance? FieldPlaceholder v)
                (instance? ExpressionRef v))
     v
@@ -379,12 +415,26 @@
 ;;; ## source-table
 
 (s/defn ^:ql ^:always-validate source-table
-  "Specify the ID of the table to query (required).
+  "Specify the ID of the table to query.
+   Queries must specify *either* `:source-table` or `:source-query`.
 
      (source-table {} 100)"
   [query, table-id :- s/Int]
   (assoc query :source-table table-id))
 
+(declare expand-inner)
+
+(s/defn ^:ql ^:always-validate source-query
+  "Specify a query to use as the source for this query (e.g., as a `SUBSELECT`).
+   Queries must specify *either* `:source-table` or `:source-query`.
+
+     (source-query {} (-> (source-table {} 100)
+                          (limit 10)))"
+  {:added "0.25.0"}
+  [query, source-query :- su/Map]
+  (assoc query :source-query (if (:native source-query)
+                               source-query
+                               (expand-inner source-query))))
 
 
 ;;; ## calculated columns
diff --git a/src/metabase/query_processor/interface.clj b/src/metabase/query_processor/interface.clj
index dfa1954c2baa1d7c1db7e72e9510f7f1d7c48f9a..eb9f4be1ef2db9ce375cd3f36e2d531040d936ee 100644
--- a/src/metabase/query_processor/interface.clj
+++ b/src/metabase/query_processor/interface.clj
@@ -75,8 +75,10 @@
 (defprotocol IField
   "Methods specific to the Query Expander `Field` record type."
   (qualified-name-components [this]
-    "Return a vector of name components of the form `[table-name parent-names... field-name]`"))
-
+    "Return a vector of name components of the form `[table-name parent-names... field-name]`
+     (This should always return AT LEAST 2 components. If no table name should be used, return
+     `nil` as the first part.)"))
+;; TODO - Yes, I know, that makes no sense. `annotate/qualify-field-name` expects it that way tho
 
 ;;; +----------------------------------------------------------------------------------------------------------------------------------------------------------------+
 ;;; |                                                                             FIELDS                                                                             |
@@ -104,7 +106,7 @@
   (getName [_] field-name) ; (name <field>) returns the *unqualified* name of the field, #obvi
 
   IField
-  (qualified-name-components [this]
+  (qualified-name-components [_]
     (conj (if parent
             (qualified-name-components parent)
             [table-name])
@@ -134,9 +136,17 @@
   [unit]
   (contains? relative-datetime-value-units (keyword unit)))
 
+;; TODO - maybe we should figure out some way to have the schema validate that the driver supports field literals, like we do for some of the other clauses.
+;; Ideally we'd do that in a more generic way (perhaps in expand, we could make the clauses specify required feature metadata and have that get checked automatically?)
+(s/defrecord FieldLiteral [field-name    :- su/NonBlankString
+                           base-type     :- su/FieldType]
+  clojure.lang.Named
+  (getName [_] field-name)
+  IField
+  (qualified-name-components [_] [nil field-name]))
 
 ;; DateTimeField is just a simple wrapper around Field
-(s/defrecord DateTimeField [field :- Field
+(s/defrecord DateTimeField [field :- (s/cond-pre Field FieldLiteral)
                             unit  :- DatetimeFieldUnit]
   clojure.lang.Named
   (getName [_] (name field)))
@@ -169,21 +179,13 @@
                                fk-field-id      :- (s/maybe (s/constrained su/IntGreaterThanZero
                                                                         (fn [_] (or (assert-driver-supports :foreign-keys) true)) ; assert-driver-supports will throw Exception if driver is bound
                                                                         "foreign-keys is not supported by this driver."))         ; and driver does not support foreign keys
-                               datetime-unit    :- (s/maybe (apply s/enum datetime-field-units))
+                               datetime-unit    :- (s/maybe DatetimeFieldUnit)
                                binning-strategy :- (s/maybe (apply s/enum binning-strategies))
                                binning-param    :- (s/maybe s/Num)])
 
 (s/defrecord AgFieldRef [index :- s/Int])
 ;; TODO - add a method to get matching expression from the query?
 
-
-
-
-(def FieldPlaceholderOrExpressionRef
-  "Schema for either a `FieldPlaceholder` or `ExpressionRef`."
-  (s/named (s/cond-pre FieldPlaceholder ExpressionRef)
-           "Valid field or expression reference."))
-
 (s/defrecord RelativeDatetime [amount :- s/Int
                                unit   :- DatetimeValueUnit])
 
@@ -199,9 +201,11 @@
 
 
 (def AnyField
-  "Schema for a anything that is considered a valid 'field'."
+  "Schema for anything that is considered a valid 'field' including placeholders, expressions, and literals."
   (s/named (s/cond-pre Field
                        FieldPlaceholder
+                       DateTimeField
+                       FieldLiteral
                        AgFieldRef
                        Expression
                        ExpressionRef)
@@ -224,13 +228,13 @@
   "Schema for an MBQL datetime value: an ISO-8601 string, `java.sql.Date`, or a relative dateitme form."
   (s/named (s/cond-pre RelativeDatetime LiteralDatetime) "Valid datetime (must ISO-8601 string literal or a relative-datetime form)"))
 
-(def OrderableValue
+(def OrderableValueLiteral
   "Schema for something that is orderable value in MBQL (either a number or datetime)."
   (s/named (s/cond-pre s/Num Datetime) "Valid orderable value (must be number or datetime)"))
 
 (def AnyValueLiteral
   "Schema for anything that is a considered a valid value literal in MBQL - `nil`, a `Boolean`, `Number`, `String`, or relative datetime form."
-  (s/named (s/maybe (s/cond-pre s/Bool su/NonBlankString OrderableValue)) "Valid value (must be nil, boolean, number, string, or a relative-datetime form)"))
+  (s/named (s/maybe (s/cond-pre s/Bool su/NonBlankString OrderableValueLiteral)) "Valid value (must be nil, boolean, number, string, or a relative-datetime form)"))
 
 
 ;; Value is the expansion of a value within a QL clause
@@ -239,13 +243,28 @@
 (s/defrecord Value [value   :- AnyValueLiteral
                     field   :- (s/recursive #'AnyField)])
 
+(s/defrecord RelativeDateTimeValue [amount :- s/Int
+                                    unit   :- DatetimeValueUnit
+                                    field  :- (s/cond-pre DateTimeField
+                                                          FieldPlaceholder)])
+
 ;; e.g. an absolute point in time (literal)
 (s/defrecord DateTimeValue [value :- Timestamp
                             field :- DateTimeField])
 
-(s/defrecord RelativeDateTimeValue [amount :- s/Int
-                                    unit   :- DatetimeValueUnit
-                                    field  :- DateTimeField])
+(def OrderableValue
+  "Schema for an instance of `Value` whose `:value` property is itself orderable (a datetime or number, i.e. a `OrderableValueLiteral`)."
+  (s/named (s/cond-pre
+            DateTimeValue
+            RelativeDateTimeValue
+            (s/constrained Value (fn [{value :value}]
+                                   (nil? (s/check OrderableValueLiteral value)))))
+           "Value that is orderable (Value whose :value is something orderable, like a datetime or number)"))
+
+(def StringValue
+  "Schema for an instance of `Value` whose `:value` property is itself a string (a datetime or string, i.e. a `OrderableValueLiteral`)."
+  (s/named (s/constrained Value (comp string? :value))
+           "Value that is a string (Value whose :value is a string)"))
 
 (defprotocol ^:private IDateTimeValue
   (unit [this]
@@ -268,20 +287,34 @@
 
 ;; Replace values with these during first pass over Query.
 ;; Include associated Field ID so appropriate the info can be found during Field resolution
-(s/defrecord ValuePlaceholder [field-placeholder :- FieldPlaceholderOrExpressionRef
+(s/defrecord ValuePlaceholder [field-placeholder :- AnyField
                                value             :- AnyValueLiteral])
 
 (def OrderableValuePlaceholder
   "`ValuePlaceholder` schema with the additional constraint that the value be orderable (a number or datetime)."
-  (s/constrained ValuePlaceholder (comp (complement (s/checker OrderableValue)) :value) ":value must be orderable (number or datetime)"))
+  (s/constrained ValuePlaceholder (comp (complement (s/checker OrderableValueLiteral)) :value) ":value must be orderable (number or datetime)"))
+
+(def OrderableValueOrPlaceholder
+  "Schema for an `OrderableValue` (instance of `Value` whose `:value` is orderable) or a placeholder for one."
+  (s/named (s/cond-pre OrderableValue OrderableValuePlaceholder)
+           "Must be an OrderableValue or OrderableValuePlaceholder"))
 
 (def StringValuePlaceholder
   "`ValuePlaceholder` schema with the additional constraint that the value be a string/"
   (s/constrained ValuePlaceholder (comp string? :value) ":value must be a string"))
 
+(def StringValueOrPlaceholder
+  "Schema for an `StringValue` (instance of `Value` whose `:value` is a string) or a placeholder for one."
+  (s/named (s/cond-pre StringValue StringValuePlaceholder)
+           "Must be an StringValue or StringValuePlaceholder"))
+
+(def AnyValue
+  "Schema that accepts anything normally considered a value or value placeholder."
+  (s/named (s/cond-pre DateTimeValue RelativeDateTimeValue Value ValuePlaceholder) "Valid value"))
+
 (def AnyFieldOrValue
-  "Schema that accepts anything normally considered a field (including expressions and literals) *or* a value or value placehoder."
-  (s/named (s/cond-pre AnyField Value ValuePlaceholder) "Field or value"))
+  "Schema that accepts anything normally considered a field or value."
+  (s/named (s/cond-pre AnyField AnyValue) "Field or value"))
 
 
 ;;; +----------------------------------------------------------------------------------------------------------------------------------------------------------------+
@@ -296,7 +329,7 @@
 
 (s/defrecord AggregationWithField [aggregation-type :- (s/named (s/enum :avg :count :cumulative-sum :distinct :max :min :stddev :sum)
                                                                 "Valid aggregation type")
-                                   field            :- (s/cond-pre FieldPlaceholderOrExpressionRef
+                                   field            :- (s/cond-pre AnyField
                                                                    Expression)
                                    custom-name      :- (s/maybe su/NonBlankString)])
 
@@ -316,21 +349,21 @@
 ;;; filter
 
 (s/defrecord EqualityFilter [filter-type :- (s/enum := :!=)
-                             field       :- FieldPlaceholderOrExpressionRef
+                             field       :- AnyField
                              value       :- AnyFieldOrValue])
 
 (s/defrecord ComparisonFilter [filter-type :- (s/enum :< :<= :> :>=)
-                               field       :- FieldPlaceholderOrExpressionRef
-                               value       :- OrderableValuePlaceholder])
+                               field       :- AnyField
+                               value       :- OrderableValueOrPlaceholder])
 
 (s/defrecord BetweenFilter [filter-type  :- (s/eq :between)
-                            min-val      :- OrderableValuePlaceholder
-                            field        :- FieldPlaceholderOrExpressionRef
-                            max-val      :- OrderableValuePlaceholder])
+                            min-val      :- OrderableValueOrPlaceholder
+                            field        :- AnyField
+                            max-val      :- OrderableValueOrPlaceholder])
 
 (s/defrecord StringFilter [filter-type :- (s/enum :starts-with :contains :ends-with)
-                           field       :- FieldPlaceholderOrExpressionRef
-                           value       :- StringValuePlaceholder])
+                           field       :- AnyField
+                           value       :- (s/cond-pre s/Str StringValueOrPlaceholder)]) ; TODO - not 100% sure why this is also allowed to accept a plain string
 
 (def SimpleFilterClause
   "Schema for a non-compound, non-`not` MBQL `filter` clause."
@@ -373,18 +406,39 @@
            "Valid page clause"))
 
 
+;;; source-query
+
+(declare Query)
+
+(def SourceQuery
+  "Schema for a valid value for a `:source-query` clause."
+  (s/if :native
+    {:native                         s/Any
+     (s/optional-key :template_tags) s/Any}
+    (s/recursive #'Query)))
+
 ;;; +----------------------------------------------------------------------------------------------------------------------------------------------------------------+
 ;;; |                                                                             QUERY                                                                              |
 ;;; +----------------------------------------------------------------------------------------------------------------------------------------------------------------+
 
 (def Query
   "Schema for an MBQL query."
-  {(s/optional-key :aggregation) [Aggregation]
-   (s/optional-key :breakout)    [FieldPlaceholderOrExpressionRef]
-   (s/optional-key :fields)      [AnyField]
-   (s/optional-key :filter)      Filter
-   (s/optional-key :limit)       su/IntGreaterThanZero
-   (s/optional-key :order-by)    [OrderBy]
-   (s/optional-key :page)        Page
-   (s/optional-key :expressions) {s/Keyword Expression}
-   :source-table                 su/IntGreaterThanZero})
+  (s/constrained
+   {(s/optional-key :aggregation)  [Aggregation]
+    (s/optional-key :breakout)     [AnyField]
+    (s/optional-key :fields)       [AnyField]
+    (s/optional-key :filter)       Filter
+    (s/optional-key :limit)        su/IntGreaterThanZero
+    (s/optional-key :order-by)     [OrderBy]
+    (s/optional-key :page)         Page
+    (s/optional-key :expressions)  {s/Keyword Expression}
+    (s/optional-key :source-table) su/IntGreaterThanZero
+    (s/optional-key :source-query) SourceQuery}
+   (fn [{:keys [source-table source-query native-source-query]}]
+     (and (or source-table
+              source-query
+              native-source-query)
+          (not (and source-table
+                    source-query
+                    native-source-query))))
+   "Query must specify either `:source-table` or `:source-query`, but not both."))
diff --git a/src/metabase/query_processor/middleware/add_implicit_clauses.clj b/src/metabase/query_processor/middleware/add_implicit_clauses.clj
index 705a461061f3224a9bf9bb770ea26ed6337fe64e..21483b4ca8499cce00e6137be0994fbaaf8d222a 100644
--- a/src/metabase/query_processor/middleware/add_implicit_clauses.clj
+++ b/src/metabase/query_processor/middleware/add_implicit_clauses.clj
@@ -5,60 +5,74 @@
             [metabase.query-processor
              [interface :as i]
              [resolve :as resolve]
+             [sort :as sort]
              [util :as qputil]]
             [toucan.db :as db]))
 
+(defn- fetch-fields-for-souce-table-id [source-table-id]
+  (map resolve/rename-mb-field-keys
+       (db/select [Field :name :display_name :base_type :special_type :visibility_type :table_id :id :position :description]
+         :table_id        source-table-id
+         :visibility_type [:not-in ["sensitive" "retired"]]
+         :parent_id       nil
+         {:order-by [[:position :asc]
+                     [:id :desc]]})))
+
 (defn- fields-for-source-table
   "Return the all fields for SOURCE-TABLE, for use as an implicit `:fields` clause."
-  [{source-table-id :id, :as source-table}]
-  (for [field (db/select [Field :name :display_name :base_type :special_type :visibility_type :table_id :id :position :description]
-                :table_id        source-table-id
-                :visibility_type [:not-in ["sensitive" "retired"]]
-                :parent_id       nil
-                {:order-by [[:position :asc]
-                            [:id :desc]]})]
-    (let [field (resolve/resolve-table (i/map->Field (resolve/rename-mb-field-keys field))
-                                       {[nil source-table-id] source-table})]
-      (if (qputil/datetime-field? field)
-        (i/map->DateTimeField {:field field, :unit :default})
-        field))))
+  [{{source-table-id :id, :as source-table} :source-table, :as inner-query}]
+  ;; Sort the implicit FIELDS so the SQL (or other native query) that gets generated (mostly) approximates the 'magic' sorting
+  ;; we do on the results. This is done so when the outer query we generate is a `SELECT *` the order doesn't change
+  (for [field (sort/sort-fields inner-query (fetch-fields-for-souce-table-id source-table-id))
+        :let  [field (resolve/resolve-table (i/map->Field field) {[nil source-table-id] source-table})]]
+    (if (qputil/datetime-field? field)
+      (i/map->DateTimeField {:field field, :unit :default})
+      field)))
 
-(defn- should-add-implicit-fields? [{{:keys [fields breakout], aggregations :aggregation} :query, :as query}]
-  (and (qputil/mbql-query? query)
+(defn- should-add-implicit-fields? [{:keys [fields breakout source-table], aggregations :aggregation}]
+  (and source-table ; if query is using another query as its source then there will be no table to add nested fields for
        (not (or (seq aggregations)
                 (seq breakout)
                 (seq fields)))))
 
-(defn- add-implicit-fields [{{:keys [source-table]} :query, :as query}]
-  (if-not (should-add-implicit-fields? query)
-    query
+(defn- add-implicit-fields [{:keys [source-table], :as inner-query}]
+  (if-not (should-add-implicit-fields? inner-query)
+    inner-query
     ;; this is a structured `:rows` query, so lets add a `:fields` clause with all fields from the source table + expressions
-    (let [fields      (fields-for-source-table source-table)
-          expressions (for [[expression-name] (get-in query [:query :expressions])]
+    (let [inner-query (assoc inner-query :fields-is-implicit true)
+          fields      (fields-for-source-table inner-query)
+          expressions (for [[expression-name] (:expressions inner-query)]
                         (i/strict-map->ExpressionRef {:expression-name (name expression-name)}))]
       (when-not (seq fields)
         (log/warn (format "Table '%s' has no Fields associated with it." (:name source-table))))
-      (-> query
-          (assoc-in [:query :fields-is-implicit] true)
-          (assoc-in [:query :fields] (concat fields expressions))))))
+      (assoc inner-query
+        :fields (concat fields expressions)))))
 
 
 
 (defn- add-implicit-breakout-order-by
   "`Fields` specified in `breakout` should add an implicit ascending `order-by` subclause *unless* that field is *explicitly* referenced in `order-by`."
-  [{{breakout-fields :breakout, order-by :order-by} :query, :as query}]
+  [{breakout-fields :breakout, order-by :order-by, :as inner-query}]
+  (let [order-by-fields                   (set (map :field order-by))
+        implicit-breakout-order-by-fields (filter (partial (complement contains?) order-by-fields)
+                                                  breakout-fields)]
+    (cond-> inner-query
+      (seq implicit-breakout-order-by-fields) (update :order-by concat (for [field implicit-breakout-order-by-fields]
+                                                                         {:field field, :direction :ascending})))))
+
+(defn- add-implicit-clauses-to-inner-query [inner-query]
+  (cond-> (add-implicit-fields (add-implicit-breakout-order-by inner-query))
+    ;; if query has a source query recursively add implicit clauses to that too as needed
+    (:source-query inner-query) (update :source-query add-implicit-clauses-to-inner-query)))
+
+(defn- maybe-add-implicit-clauses [query]
   (if-not (qputil/mbql-query? query)
     query
-    (let [order-by-fields                   (set (map :field order-by))
-          implicit-breakout-order-by-fields (filter (partial (complement contains?) order-by-fields)
-                                                    breakout-fields)]
-      (cond-> query
-        (seq implicit-breakout-order-by-fields) (update-in [:query :order-by] concat (for [field implicit-breakout-order-by-fields]
-                                                                                       {:field field, :direction :ascending}))))))
+    (update query :query add-implicit-clauses-to-inner-query)))
 
 
 (defn add-implicit-clauses
   "Add an implicit `fields` clause to queries with no `:aggregation`, `breakout`, or explicit `:fields` clauses.
    Add implicit `:order-by` clauses for fields specified in a `:breakout`."
   [qp]
-  (comp qp add-implicit-fields add-implicit-breakout-order-by))
+  (comp qp maybe-add-implicit-clauses))
diff --git a/src/metabase/query_processor/middleware/cache.clj b/src/metabase/query_processor/middleware/cache.clj
index de1c58bc69278932f4a20aa8ee413498c0ee8940..384d18f69636f933f47413c131cc943cbd91e5e9 100644
--- a/src/metabase/query_processor/middleware/cache.clj
+++ b/src/metabase/query_processor/middleware/cache.clj
@@ -58,10 +58,10 @@
   ([]
    (set-backend! (config/config-kw :mb-qp-cache-backend)))
   ([backend]
-   (let [backend-ns (symbol (str "metabase.query-processor.middleware.cache-backend." (munge (name backend))))]
-     (require backend-ns)
+   (let [backend-ns-symb (symbol (str "metabase.query-processor.middleware.cache-backend." (munge (name backend))))]
+     (require backend-ns-symb)
      (log/info "Using query processor cache backend:" (u/format-color 'blue backend) (u/emoji "💾"))
-     (reset! backend-instance (get-backend-instance-in-namespace backend-ns)))))
+     (reset! backend-instance (get-backend-instance-in-namespace backend-ns-symb)))))
 
 
 
diff --git a/src/metabase/query_processor/middleware/fetch_source_query.clj b/src/metabase/query_processor/middleware/fetch_source_query.clj
new file mode 100644
index 0000000000000000000000000000000000000000..fc3e6eeaceea475203c67bf178a85656bce0ccb2
--- /dev/null
+++ b/src/metabase/query_processor/middleware/fetch_source_query.clj
@@ -0,0 +1,61 @@
+(ns metabase.query-processor.middleware.fetch-source-query
+  (:require [clojure.tools.logging :as log]
+            [metabase.query-processor
+             [interface :as i]
+             [util :as qputil]]
+            [metabase.util :as u]
+            [toucan.db :as db]))
+
+(defn- card-id->source-query
+  "Return the source query info for Card with CARD-ID."
+  [card-id]
+  (let [card       (db/select-one ['Card :dataset_query :database_id] :id card-id)
+        card-query (:dataset_query card)]
+    (assoc (or (:query card-query)
+               (when-let [native (:native card-query)]
+                 {:native        (:query native)
+                  :template_tags (:template_tags native)})
+               (throw (Exception. (str "Missing source query in Card " card-id))))
+      ;; include database ID as well; we'll pass that up the chain so it eventually gets put in its spot in the outer-query
+      :database (:database card-query))))
+
+(defn- source-table-str->source-query
+  "Given a SOURCE-TABLE-STR like `card__100` return the appropriate source query."
+  [source-table-str]
+  (let [[_ card-id-str] (re-find #"^card__(\d+)$" source-table-str)]
+    (u/prog1 (card-id->source-query (Integer/parseInt card-id-str))
+      (when-not i/*disable-qp-logging*
+        (log/info "\nFETCHED SOURCE QUERY FROM CARD" card-id-str ":\n" (u/pprint-to-str 'yellow <>))))))
+
+(defn- expand-card-source-tables
+  "If `source-table` is a Card reference (a string like `card__100`) then replace that with appropriate `:source-query` information.
+   Does nothing if `source-table` is a normal ID. Recurses for nested-nested queries."
+  [inner-query]
+  (let [source-table (qputil/get-normalized inner-query :source-table)]
+    (if-not (string? source-table)
+      inner-query
+      ;; (recursively) expand the source query
+      (let [source-query (expand-card-source-tables (source-table-str->source-query source-table))]
+        (-> inner-query
+            ;; remove `source-table` `card__id` key
+            (qputil/dissoc-normalized :source-table)
+            ;; Add new `source-query` info in its place. Pass the database ID up the chain, removing it from the source query
+            (assoc
+              :source-query (dissoc source-query :database)
+              :database     (:database source-query)))))))
+
+(defn- fetch-source-query* [{inner-query :query, :as outer-query}]
+  (if-not inner-query
+    ;; for non-MBQL queries there's nothing to do since they have nested queries
+    outer-query
+    ;; otherwise attempt to expand any source queries as needed
+    (let [expanded-inner-query (expand-card-source-tables inner-query)]
+      (merge outer-query
+             {:query (dissoc expanded-inner-query :database)}
+             (when-let [database (:database expanded-inner-query)]
+               {:database database})))))
+
+(defn fetch-source-query
+  "Middleware that assocs the `:source-query` for this query if it was specified using the shorthand `:source-table` `card__n` format."
+  [qp]
+  (comp qp fetch-source-query*))
diff --git a/src/metabase/query_processor/middleware/log.clj b/src/metabase/query_processor/middleware/log.clj
index 6ac5031e61c05de6f50fbaa8a26f7abf72efe870..183668ba9705e625ba3cd1ae21c80d3a404efdc1 100644
--- a/src/metabase/query_processor/middleware/log.clj
+++ b/src/metabase/query_processor/middleware/log.clj
@@ -20,7 +20,7 @@
                     (walk/prewalk
                      (fn [f]
                        (if-not (map? f) f
-                               (m/filter-vals identity (into {} f))))
+                               (m/filter-vals (complement nil?) (into {} f))))
                      ;; obscure DB details when logging. Just log the name of driver because we don't care about its properties
                      (-> query
                          (assoc-in [:database :details] (u/emoji "😋 ")) ; :yum:
@@ -42,3 +42,16 @@
   "Middleware for logging a query when it is very first encountered, before it is expanded."
   [qp]
   (comp qp log-initial-query*))
+
+
+(defn- log-results-metadata* [results]
+  (u/prog1 results
+    (when-not i/*disable-qp-logging*
+      (log/debug "Result Metadata:\n"
+                 (u/pprint-to-str 'blue (for [col (get-in <> [:data :cols])]
+                                          (m/filter-vals (complement nil?) col)))))))
+
+(defn log-results-metadata
+  "Middleware that logs the column metadata that comes back with the results."
+  [qp]
+  (comp log-results-metadata* qp))
diff --git a/src/metabase/query_processor/middleware/parameters.clj b/src/metabase/query_processor/middleware/parameters.clj
index 53d25787bb1c887e2257f4d4b3c7dea5878bdf85..f6d013ca8835d96a8cb69fb6316db395ee26e405 100644
--- a/src/metabase/query_processor/middleware/parameters.clj
+++ b/src/metabase/query_processor/middleware/parameters.clj
@@ -2,21 +2,49 @@
   "Middleware for substituting parameters in queries."
   (:require [clojure.data :as data]
             [clojure.tools.logging :as log]
-            [metabase.query-processor.interface :as i]
+            [metabase.driver.generic-sql.util.unprepare :as unprepare]
+            [metabase.query-processor
+             [interface :as i]
+             [util :as qputil]]
             [metabase.query-processor.middleware.parameters
              [mbql :as mbql-params]
              [sql :as sql-params]]
             [metabase.util :as u]))
 
-(defn- expand-parameters
+(defn- expand-parameters*
   "Expand any :parameters set on the QUERY-DICT and apply them to the query definition.
    This function removes the :parameters attribute from the QUERY-DICT as part of its execution."
   [{:keys [parameters], :as query-dict}]
   ;; params in native queries are currently only supported for SQL drivers
-  (if (= :query (keyword (:type query-dict)))
+  (if (qputil/mbql-query? query-dict)
     (mbql-params/expand (dissoc query-dict :parameters) parameters)
     (sql-params/expand query-dict)))
 
+(defn- expand-params-in-native-source-query
+  "Expand parameters in a native source query."
+  [{{{original-query :native, tags :template_tags} :source-query} :query, :as outer-query}]
+  ;; TODO - This isn't recursive for nested-nested queries
+  ;; TODO - Yes, this approach is hacky. But hacky & working > not working
+  (let [{{new-query :query, new-params :params} :native} (sql-params/expand (assoc outer-query
+                                                                              :type   :native
+                                                                              :native {:query         original-query
+                                                                                       :template_tags tags}))]
+    (if (= original-query new-query)
+      ;; if the native query didn't change, we don't need to do anything; return as-is
+      outer-query
+      ;; otherwise replace the native query with the param-substituted version.
+      ;; 'Unprepare' the args because making sure args get passed in the right order is too tricky for nested queries
+      ;; TODO - This might not work for all drivers. We should make 'unprepare' a Generic SQL method
+      ;; so different drivers can invoke unprepare/unprepare with the correct args
+      (-> outer-query
+          (assoc-in [:query :source-query :native] (unprepare/unprepare (cons new-query new-params)))))))
+
+(defn- expand-parameters
+  "Expand parameters in the OUTER-QUERY, and if the query is using a native source query, expand params in that as well."
+  [outer-query]
+  (cond-> (expand-parameters* outer-query)
+    (get-in outer-query [:query :source-query :native]) expand-params-in-native-source-query))
+
 (defn- substitute-parameters*
   "If any parameters were supplied then substitute them into the query."
   [query]
diff --git a/src/metabase/query_processor/middleware/permissions.clj b/src/metabase/query_processor/middleware/permissions.clj
index fddb9f6e7a491074e95259c186310c17d429a036..2fdf81f66e30dc3a419f9d5596a255f347e5682b 100644
--- a/src/metabase/query_processor/middleware/permissions.clj
+++ b/src/metabase/query_processor/middleware/permissions.clj
@@ -79,9 +79,11 @@
 ;; TODO - why is this the only function here that takes `user-id`?
 (defn- throw-if-cannot-run-query
   "Throw an exception if USER-ID doesn't have permissions to run QUERY."
-  [user-id {:keys [source-table join-tables]}]
-  (doseq [table (cons source-table join-tables)]
-    (throw-if-cannot-run-query-referencing-table user-id table)))
+  [user-id {:keys [source-table join-tables source-query]}]
+  (if source-query
+    (recur user-id source-query)
+    (doseq [table (cons source-table join-tables)]
+      (throw-if-cannot-run-query-referencing-table user-id table))))
 
 
 ;;; ------------------------------------------------------------ Permissions for Native Queries ------------------------------------------------------------
@@ -120,7 +122,7 @@
 
 (defn- check-query-permissions-for-user
   "Check that User with USER-ID has permissions to run QUERY, or throw an exception."
-  [user-id {query-type :type, database :database, query :query, {card-id :card-id} :info, :as outer-query}]
+  [user-id {query-type :type, database :database, {:keys [source-query], :as query} :query, {:keys [card-id]} :info, :as outer-query}]
   {:pre [(integer? user-id) (map? outer-query)]}
   (let [native?       (= (keyword query-type) :native)
         collection-id (db/select-one-field :collection_id 'Card :id card-id)]
@@ -128,6 +130,11 @@
       ;; if the card is in a COLLECTION, then see if the current user has permissions for that collection
       collection-id
       (throw-if-user-doesnt-have-access-to-collection collection-id)
+      ;; Otherwise if this is a NESTED query then we should check permissions for the source query
+      source-query
+      (if (:native source-query)
+        (throw-if-cannot-run-existing-native-query-referencing-db database)
+        (throw-if-cannot-run-query user-id source-query))
       ;; for native queries that are *not* part of an existing card, check that we have native permissions for the DB
       (and native? (not card-id))
       (throw-if-cannot-run-new-native-query-referencing-db database)
diff --git a/src/metabase/query_processor/middleware/results_metadata.clj b/src/metabase/query_processor/middleware/results_metadata.clj
new file mode 100644
index 0000000000000000000000000000000000000000..824b54e0a22a97fd20d3d93d3ef3013b8cd05ae3
--- /dev/null
+++ b/src/metabase/query_processor/middleware/results_metadata.clj
@@ -0,0 +1,98 @@
+(ns metabase.query-processor.middleware.results-metadata
+  "Middleware that stores metadata about results column types after running a query for a Card,
+   and returns that metadata (which can be passed *back* to the backend when saving a Card) as well
+   as a checksum in the API response."
+  (:require [buddy.core.hash :as hash]
+            [cheshire.core :as json]
+            [metabase.models.humanization :as humanization]
+            [metabase.query-processor.interface :as i]
+            [metabase.util :as u]
+            [metabase.util
+             [encryption :as encryption]
+             [schema :as su]]
+            [ring.util.codec :as codec]
+            [schema.core :as s]
+            [toucan.db :as db]))
+
+(def ^:private DateTimeUnitKeywordOrString
+  "Schema for a valid datetime unit string like \"default\" or \"minute-of-hour\"."
+  (s/constrained su/KeywordOrString
+                 (fn [unit]
+                   (contains? i/datetime-field-units (keyword unit)))
+                 "Valid field datetime unit keyword or string"))
+
+(def ResultsMetadata
+  "Schema for valid values of the `result_metadata` column."
+  (su/with-api-error-message (s/named [{:name                          su/NonBlankString
+                                        :display_name                  su/NonBlankString
+                                        (s/optional-key :description)  (s/maybe su/NonBlankString)
+                                        :base_type                     su/FieldTypeKeywordOrString
+                                        (s/optional-key :special_type) (s/maybe su/FieldTypeKeywordOrString)
+                                        (s/optional-key :unit)         (s/maybe DateTimeUnitKeywordOrString)}]
+                                      "Valid array of results column metadata maps")
+    "value must be an array of valid results column metadata maps."))
+
+(s/defn ^:private ^:always-validate results->column-metadata :- (s/maybe ResultsMetadata)
+  "Return the desired storage format for the column metadata coming back from RESULTS, or `nil` if no columns were returned."
+  [results]
+  ;; rarely certain queries will return columns with no names, for example `SELECT COUNT(*)` in SQL Server seems to come back with no name
+  ;; since we can't use those as field literals in subsequent queries just filter them out
+  (seq (for [col   (:cols results)
+             :when (seq (:name col))]
+         (merge
+          ;; if base-type isn't set put a default one in there. Similarly just use humanized value of `:name` for `:display_name` if one isn't set
+          {:base_type    :type/*
+           :display_name (humanization/name->human-readable-name (name (:name col)))}
+          (u/select-non-nil-keys col [:name :display_name :description :base_type :special_type :unit])
+          ;; since years are actually returned as text they can't be used for breakout purposes so don't advertise them as DateTime columns
+          (when (= (:unit col) :year)
+            {:base_type :type/Text
+             :unit      nil})))))
+
+;; TODO - is there some way we could avoid doing this every single time a Card is ran? Perhaps by passing the current Card
+;; metadata as part of the query context so we can compare for changes
+(defn- record-metadata! [card-id metadata]
+  (when metadata
+    (db/update! 'Card card-id
+      :result_metadata metadata)))
+
+(defn- metadata-checksum
+  "Simple, checksum of the column results METADATA.
+   Results metadata is returned as part of all query results, with the hope that the frontend will pass it back to
+   us when a Card is saved or updated. This checksum (also passed) is a simple way for us to check whether the metadata
+   is valid and hasn't been accidentally tampered with.
+
+   By default, this is not cryptographically secure, nor is it meant to be. Of course, a bad actor could alter the
+   metadata and return a new, correct checksum. But intentionally saving bad metadata would only help in letting you
+   write bad queries; the field literals can only refer to columns in the original 'source' query at any rate, so you
+   wouldn't, for example, be able to give yourself access to columns in a different table.
+
+   However, if `MB_ENCRYPTION_SECRET_KEY` is set, we'll go ahead and use it to encypt the checksum so it becomes it becomes
+   impossible to alter the metadata and produce a correct checksum at any rate."
+  [metadata]
+  (when metadata
+    (encryption/maybe-encrypt (codec/base64-encode (hash/md5 (json/generate-string metadata))))))
+
+(defn valid-checksum?
+  "Is the CHECKSUM the right one for this column METADATA?"
+  [metadata checksum]
+  (and metadata
+       checksum
+       (= (encryption/maybe-decrypt (metadata-checksum metadata))
+          (encryption/maybe-decrypt checksum))))
+
+(defn record-and-return-metadata!
+  "Middleware that records metadata about the columns returned when running the query if it is associated with a Card."
+  [qp]
+  (fn [{{:keys [card-id nested?]} :info, :as query}]
+    (let [results  (qp query)
+          metadata (results->column-metadata results)]
+      ;; At the very least we can skip the Extra DB call to update this Card's metadata results
+      ;; if its DB doesn't support nested queries in the first place
+      (when (i/driver-supports? :nested-queries)
+        (when (and card-id
+                   (not nested?))
+          (record-metadata! card-id metadata)))
+      ;; add the metadata and checksum to the response
+      (assoc results :results_metadata {:checksum (metadata-checksum metadata)
+                                        :columns  metadata}))))
diff --git a/src/metabase/query_processor/resolve.clj b/src/metabase/query_processor/resolve.clj
index f55fcc9fee220995076c495fe20530e6bd0cd001..900e1b754d976eebcaec91e9481e4d3a7101b751 100644
--- a/src/metabase/query_processor/resolve.clj
+++ b/src/metabase/query_processor/resolve.clj
@@ -266,19 +266,25 @@
 (defn- resolve-tables
   "Resolve the `Tables` in an EXPANDED-QUERY-DICT."
   [{{source-table-id :source-table} :query, :keys [table-ids fk-field-ids], :as expanded-query-dict}]
-  {:pre [(integer? source-table-id)]}
-  (let [table-ids             (conj table-ids source-table-id)
-        source-table          (or (db/select-one [Table :schema :name :id], :id source-table-id)
-                                  (throw (Exception. (format "Query expansion failed: could not find source table %d." source-table-id))))
-        joined-tables         (fk-field-ids->joined-tables source-table-id fk-field-ids)
-        fk-id+table-id->table (into {[nil source-table-id] source-table}
-                                    (for [{:keys [source-field table-id join-alias]} joined-tables]
-                                      {[(:field-id source-field) table-id] {:name join-alias
-                                                                            :id   table-id}}))]
-    (as-> expanded-query-dict <>
-      (assoc-in <> [:query :source-table] source-table)
-      (assoc-in <> [:query :join-tables]  joined-tables)
-      (walk/postwalk #(resolve-table % fk-id+table-id->table) <>))))
+  (if-not source-table-id
+    ;; if we have a `source-query`, recurse and resolve tables in that
+    (update-in expanded-query-dict [:query :source-query] (fn [source-query]
+                                                            (if (:native source-query)
+                                                              source-query
+                                                              (:query (resolve-tables (assoc expanded-query-dict :query source-query))))))
+    ;; otherwise we can resolve tables in the (current) top-level
+    (let [table-ids             (conj table-ids source-table-id)
+          source-table          (or (db/select-one [Table :schema :name :id], :id source-table-id)
+                                    (throw (Exception. (format "Query expansion failed: could not find source table %d." source-table-id))))
+          joined-tables         (fk-field-ids->joined-tables source-table-id fk-field-ids)
+          fk-id+table-id->table (into {[nil source-table-id] source-table}
+                                      (for [{:keys [source-field table-id join-alias]} joined-tables]
+                                        {[(:field-id source-field) table-id] {:name join-alias
+                                                                              :id   table-id}}))]
+      (as-> expanded-query-dict <>
+        (assoc-in <> [:query :source-table] source-table)
+        (assoc-in <> [:query :join-tables]  joined-tables)
+        (walk/postwalk #(resolve-table % fk-id+table-id->table) <>)))))
 
 
 ;;; # ------------------------------------------------------------ PUBLIC INTERFACE ------------------------------------------------------------
diff --git a/src/metabase/query_processor/sort.clj b/src/metabase/query_processor/sort.clj
index 2cf435974626f8df0c583ee47a7982d77367b117..f78153c9e12126cfdc9b8cfbec0c363d8252c742 100644
--- a/src/metabase/query_processor/sort.clj
+++ b/src/metabase/query_processor/sort.clj
@@ -65,10 +65,22 @@
        (special-type-importance field)
        field-name])))
 
+(defn- should-sort? [inner-query]
+  (or
+   ;; if there's no source query then always sort
+   (not (:source-query inner-query))
+   ;; if the source query is MBQL then sort
+   (not (get-in inner-query [:source-query :native]))
+   ;; otherwise only sort queries with *NATIVE* source queries if the query has an aggregation and/or breakout
+   (:aggregation inner-query)
+   (:breakout inner-query)))
+
 (defn sort-fields
   "Sort FIELDS by their \"importance\" vectors."
-  [query fields]
-  (let [field-importance (field-importance-fn query)]
-    (when-not i/*disable-qp-logging*
-      (log/debug (u/format-color 'yellow "Sorted fields:\n%s" (u/pprint-to-str (sort (map field-importance fields))))))
-    (sort-by field-importance fields)))
+  [inner-query fields]
+  (if-not (should-sort? inner-query)
+    fields
+    (let [field-importance (field-importance-fn inner-query)]
+      (when-not i/*disable-qp-logging*
+        (log/debug (u/format-color 'yellow "Sorted fields:\n%s" (u/pprint-to-str (sort (map field-importance fields))))))
+      (sort-by field-importance fields))))
diff --git a/src/metabase/task.clj b/src/metabase/task.clj
index 8580c03d63de8c8f0d361aeacbd007e359950e4b..5a75c79423c9a51964d1aced2f7167ddcf593b0c 100644
--- a/src/metabase/task.clj
+++ b/src/metabase/task.clj
@@ -52,3 +52,10 @@
   [job trigger]
   (when @quartz-scheduler
     (qs/schedule @quartz-scheduler job trigger)))
+
+(defn delete-task!
+  "delete a task from the scheduler"
+  [job-key trigger-key]
+  (when @quartz-scheduler
+    (qs/delete-trigger @quartz-scheduler trigger-key)
+    (qs/delete-job @quartz-scheduler job-key)))
diff --git a/src/metabase/util.clj b/src/metabase/util.clj
index 9b79d0c0302c810a830fd28021be92a1e1b541ba..36e10258a69fa49e8ec82e96f3519b75ada5deff 100644
--- a/src/metabase/util.clj
+++ b/src/metabase/util.clj
@@ -503,24 +503,29 @@
     identity
     (constantly "")))
 
-(def ^String ^{:style/indent 2, :arglists '([color-symb x] [color-symb format-str & args])}
-  format-color
+(def ^:private ^{:arglists '([color-symb x])} colorize
+  "Colorize string X with the function matching COLOR-SYMB, but only if `MB_COLORIZE_LOGS` is enabled (the default)."
+  (if (config/config-bool :mb-colorize-logs)
+    (fn [color-symb x]
+      (let [color-fn (or (ns-resolve 'colorize.core color-symb)
+                         (throw (Exception. (str "Invalid color symbol: " color-symb))))]
+        (color-fn x)))
+    (fn [_ x]
+      x)))
+
+(defn format-color
   "Like `format`, but uses a function in `colorize.core` to colorize the output.
    COLOR-SYMB should be a quoted symbol like `green`, `red`, `yellow`, `blue`,
    `cyan`, `magenta`, etc. See the entire list of avaliable colors
    [here](https://github.com/ibdknox/colorize/blob/master/src/colorize/core.clj).
 
      (format-color 'red \"Fatal error: %s\" error-message)"
-  (if (config/config-bool :mb-colorize-logs)
-    (fn
-      ([color-symb x]
-       {:pre [(symbol? color-symb)]}
-       ((ns-resolve 'colorize.core color-symb) x))
-      ([color-symb format-string & args]
-       (format-color color-symb (apply format format-string args))))
-    (fn
-      ([_ x] x)
-      ([_ format-string & args] (apply format format-string args)))))
+  {:style/indent 2}
+  (^String [color-symb x]
+   {:pre [(symbol? color-symb)]}
+   (colorize color-symb x))
+  (^String [color-symb format-string & args]
+   (colorize color-symb (apply format format-string args))))
 
 (defn pprint-to-str
   "Returns the output of pretty-printing X as a string.
@@ -528,11 +533,12 @@
    function from `colorize.core`.
 
      (pprint-to-str 'green some-obj)"
-  ([x]
+  {:style/indent 1}
+  (^String [x]
    (when x
      (with-out-str (pprint x))))
-  ([color-symb x]
-   ((ns-resolve 'colorize.core color-symb) (pprint-to-str x))))
+  (^String [color-symb x]
+   (colorize color-symb (pprint-to-str x))))
 
 (def emoji-progress-bar
   "Create a string that shows progress for something, e.g. a database sync process.
@@ -564,15 +570,39 @@
              (s/join (repeat blanks "·"))
              (format "] %s  %3.0f%%" (emoji (percent-done->emoji percent-done)) (* percent-done 100.0)))))))
 
-(defn filtered-stacktrace
-  "Get the stack trace associated with E and return it as a vector with non-metabase frames filtered out."
-  [^Throwable e]
-  (when e
-    (when-let [stacktrace (.getStackTrace e)]
-      (vec (for [frame stacktrace
-                 :let  [s (str frame)]
-                 :when (re-find #"metabase" s)]
-             (s/replace s #"^metabase\." ""))))))
+
+(defprotocol ^:private IFilteredStacktrace
+  (filtered-stacktrace [this]
+    "Get the stack trace associated with E and return it as a vector with non-metabase frames filtered out."))
+
+;; These next two functions are a workaround for this bug https://dev.clojure.org/jira/browse/CLJ-1790
+;; When Throwable/Thread are type-hinted, they return an array of type StackTraceElement, this causes
+;; a VerifyError. Adding a layer of indirection here avoids the problem. Once we upgrade to Clojure 1.9
+;; we should be able to remove this code.
+(defn- throwable-get-stack-trace [^Throwable t]
+  (.getStackTrace t))
+
+(defn- thread-get-stack-trace [^Thread t]
+  (.getStackTrace t))
+
+(extend nil
+  IFilteredStacktrace {:filtered-stacktrace (constantly nil)})
+
+(extend Throwable
+  IFilteredStacktrace {:filtered-stacktrace (fn [this]
+                                             (filtered-stacktrace (throwable-get-stack-trace this)))})
+
+(extend Thread
+  IFilteredStacktrace {:filtered-stacktrace (fn [this]
+                                              (filtered-stacktrace (thread-get-stack-trace this)))})
+
+;; StackTraceElement[] is what the `.getStackTrace` method for Thread and Throwable returns
+(extend (Class/forName "[Ljava.lang.StackTraceElement;")
+  IFilteredStacktrace {:filtered-stacktrace (fn [this]
+                                              (vec (for [frame this
+                                                         :let  [s (str frame)]
+                                                         :when (re-find #"metabase" s)]
+                                                     (s/replace s #"^metabase\." ""))))})
 
 (defn wrap-try-catch
   "Returns a new function that wraps F in a `try-catch`. When an exception is caught, it is logged
@@ -825,10 +855,10 @@
 
 (defn occurances-of-substring
   "Return the number of times SUBSTR occurs in string S."
-  ^Integer [^String s, ^String substr]
+  ^Long [^String s, ^String substr]
   (when (and (seq s) (seq substr))
     (loop [index 0, cnt 0]
-      (if-let [new-index (s/index-of s substr index)]
+      (if-let [^long new-index (s/index-of s substr index)]
         (recur (inc new-index) (inc cnt))
         cnt))))
 
diff --git a/src/metabase/util/infer_spaces.clj b/src/metabase/util/infer_spaces.clj
index 4de995ac2a0ed7f6672b1e5083eceb14345ecd39..3295c582f100784c551975951f0992f9d9b5e1e1 100644
--- a/src/metabase/util/infer_spaces.clj
+++ b/src/metabase/util/infer_spaces.clj
@@ -59,7 +59,7 @@
 ;;
 ;;     return " ".join(reversed(out))
 (defn infer-spaces
-  "Splits a string with no spaces into words using magic"
+  "Splits a string with no spaces into words using magic" ; what a great explanation. TODO - make this code readable
   [input]
   (let [s (s/lower-case input)
         cost (build-cost-array s)]
diff --git a/test/metabase/api/card_test.clj b/test/metabase/api/card_test.clj
index 0c4f47d260a5fae320702ba3a9214d4baa27bf99..1553fa6bebef29291cf8b4e457e784715dc2ba8d 100644
--- a/test/metabase/api/card_test.clj
+++ b/test/metabase/api/card_test.clj
@@ -40,7 +40,8 @@
    :made_public_by_id nil
    :public_uuid       nil
    :query_type        "query"
-   :cache_ttl         nil})
+   :cache_ttl         nil
+   :result_metadata   nil})
 
 (defn- do-with-self-cleaning-random-card-name
   "Generate a random card name (or use CARD-NAME), pass it to F, then delete any Cards with that name afterwords."
@@ -181,12 +182,11 @@
   [card-2-id]
   (map :id ((user->client :rasta) :get 200 "card", :label "more_toucans")))                 ; filtering is done by slug
 
-(defn- mbql-count-query [database-id table-id]
-  {:database database-id
-   :type     "query"
-   :query    {:source-table table-id, :aggregation {:aggregation-type "count"}}})
 
-;; ## POST /api/card
+;;; +------------------------------------------------------------------------------------------------------------------------+
+;;; |                                                    CREATING A CARD                                                     |
+;;; +------------------------------------------------------------------------------------------------------------------------+
+
 ;; Test that we can make a card
 (let [card-name (random-name)]
   (tt/expect-with-temp [Database [{database-id :id}]
@@ -199,12 +199,45 @@
             :database_id            database-id ; these should be inferred automatically
             :table_id               table-id})
     (with-self-cleaning-random-card-name [_ card-name]
-      (dissoc ((user->client :rasta) :post 200 "card" {:name                   card-name
-                                                       :display                "scalar"
-                                                       :dataset_query          (mbql-count-query database-id table-id)
-                                                       :visualization_settings {:global {:title nil}}})
+      (dissoc ((user->client :rasta) :post 200 "card" (card-with-name-and-query card-name (mbql-count-query database-id table-id)))
               :created_at :updated_at :id))))
 
+;; Make sure when saving a Card the query metadata is saved (if correct)
+(expect
+  [{:base_type    "type/Integer"
+    :display_name "Count Chocula"
+    :name         "count_chocula"
+    :special_type "type/Number"}]
+  (let [metadata [{:base_type    :type/Integer
+                   :display_name "Count Chocula"
+                   :name         "count_chocula"
+                   :special_type :type/Number}]]
+    (with-self-cleaning-random-card-name [card-name]
+      ;; create a card with the metadata
+      ((user->client :rasta) :post 200 "card" (assoc (card-with-name-and-query card-name (mbql-count-query (data/id) (data/id :venues)))
+                                                :result_metadata    metadata
+                                                :metadata_checksum  ((resolve 'metabase.query-processor.middleware.results-metadata/metadata-checksum) metadata)))
+      ;; now check the metadata that was saved in the DB
+      (db/select-one-field :result_metadata Card :name card-name))))
+
+;; make sure when saving a Card the correct query metadata is fetched (if incorrect)
+(expect
+  [{:base_type    "type/Integer"
+    :display_name "count"
+    :name         "count"
+    :special_type "type/Number"}]
+  (let [metadata [{:base_type    :type/Integer
+                   :display_name "Count Chocula"
+                   :name         "count_chocula"
+                   :special_type :type/Number}]]
+    (with-self-cleaning-random-card-name [card-name]
+      ;; create a card with the metadata
+      ((user->client :rasta) :post 200 "card" (assoc (card-with-name-and-query card-name (mbql-count-query (data/id) (data/id :venues)))
+                                                :result_metadata    metadata
+                                                :metadata_checksum  "ABCDEF")) ; bad checksum
+      ;; now check the correct metadata was fetched and was saved in the DB
+      (db/select-one-field :result_metadata Card :name card-name))))
+
 
 ;;; +------------------------------------------------------------------------------------------------------------------------+
 ;;; |                                                FETCHING A SPECIFIC CARD                                                |
@@ -323,7 +356,50 @@
     (tu/with-temporary-setting-values [enable-embedding false]
       ((user->client :crowberto) :put 400 (str "card/" (u/get-id card)) {:embedding_params {:abc "enabled"}}))))
 
-;; ## DELETE /api/card/:id
+;; make sure when updating a Card the query metadata is saved (if correct)
+(expect
+  [{:base_type    "type/Integer"
+    :display_name "Count Chocula"
+    :name         "count_chocula"
+    :special_type "type/Number"}]
+  (let [metadata [{:base_type    :type/Integer
+                   :display_name "Count Chocula"
+                   :name         "count_chocula"
+                   :special_type :type/Number}]]
+    (tt/with-temp Card [card]
+      ;; update the Card's query
+      ((user->client :rasta) :put 200 (str "card/" (u/get-id card))
+       {:dataset_query (mbql-count-query (data/id) (data/id :venues))
+        :result_metadata    metadata
+        :metadata_checksum  ((resolve 'metabase.query-processor.middleware.results-metadata/metadata-checksum) metadata)})
+      ;; now check the metadata that was saved in the DB
+      (db/select-one-field :result_metadata Card :id (u/get-id card)))))
+
+;; Make sure when updating a Card the correct query metadata is fetched (if incorrect)
+(expect
+  [{:base_type    "type/Integer"
+    :display_name "count"
+    :name         "count"
+    :special_type "type/Number"}]
+  (let [metadata [{:base_type    :type/Integer
+                   :display_name "Count Chocula"
+                   :name         "count_chocula"
+                   :special_type :type/Number}]]
+    (tt/with-temp Card [card]
+      ;; update the Card's query
+      ((user->client :rasta) :put 200 (str "card/" (u/get-id card))
+       {:dataset_query (mbql-count-query (data/id) (data/id :venues))
+        :result_metadata    metadata
+        :metadata_checksum  "ABC123"})  ; invalid checksum
+      ;; now check the metadata that was saved in the DB
+      (db/select-one-field :result_metadata Card :id (u/get-id card)))))
+
+
+;;; +------------------------------------------------------------------------------------------------------------------------+
+;;; |                                              DELETING A CARD (DEPRECATED)                                              |
+;;; +------------------------------------------------------------------------------------------------------------------------+
+;; Deprecated because you're not supposed to delete cards anymore. Archive them instead
+
 ;; Check that we can delete a card
 (expect
   nil
diff --git a/test/metabase/api/dashboard_test.clj b/test/metabase/api/dashboard_test.clj
index d2fdb6bff2bff3e7f842e658073e04e25395a5fb..ec02621cd5ee9b79cc596f2905acc3a319442218 100644
--- a/test/metabase/api/dashboard_test.clj
+++ b/test/metabase/api/dashboard_test.clj
@@ -133,7 +133,8 @@
                                                            :query_type             nil
                                                            :dataset_query          {}
                                                            :visualization_settings {}
-                                                           :query_average_duration nil})
+                                                           :query_average_duration nil
+                                                           :result_metadata        nil})
                            :series                 []}]})
   ;; fetch a dashboard WITH a dashboard card on it
   (tt/with-temp* [Dashboard     [{dashboard-id :id} {:name "Test Dashboard"}]
diff --git a/test/metabase/api/database_test.clj b/test/metabase/api/database_test.clj
index 75e5a708802e836f7206c96642ae18e740074e9f..7b99ff89b9008c2e3dbe9c6112f8b609bc586e47 100644
--- a/test/metabase/api/database_test.clj
+++ b/test/metabase/api/database_test.clj
@@ -4,7 +4,9 @@
              [driver :as driver]
              [util :as u]]
             [metabase.models
-             [database :refer [Database]]
+             [card :refer [Card]]
+             [collection :refer [Collection]]
+             [database :as database :refer [Database]]
              [field :refer [Field]]
              [table :refer [Table]]]
             [metabase.test
@@ -15,7 +17,8 @@
              [users :refer :all]]
             [toucan
              [db :as db]
-             [hydrate :as hydrate]]))
+             [hydrate :as hydrate]]
+            [toucan.util.test :as tt]))
 
 ;; HELPER FNS
 
@@ -123,11 +126,6 @@
       (dissoc (into {} (db/select-one [Database :name :engine :details :is_full_sync], :id db-id))
               :features)))
 
-:description             nil
-                               :entity_type             nil
-                               :caveats                 nil
-                               :points_of_interest      nil
-                               :visibility_type         nil
 (def ^:private default-table-details
   {:description             nil
    :entity_name             nil
@@ -319,3 +317,117 @@
   [["CATEGORIES" "Table"]
    ["CATEGORY_ID" "VENUES :type/Integer :type/FK"]]
   ((user->client :rasta) :get 200 (format "database/%d/autocomplete_suggestions" (id)) :prefix "cat"))
+
+
+;;; GET /api/database?include_cards=true
+;; Check that we get back 'virtual' tables for Saved Questions
+(defn- card-with-native-query {:style/indent 1} [card-name & {:as kvs}]
+  (merge {:name          card-name
+          :database_id   (data/id)
+          :dataset_query {:database (data/id)
+                          :type     :native
+                          :native   {:query (format "SELECT * FROM VENUES")}}}
+         kvs))
+
+(defn- card-with-mbql-query {:style/indent 1} [card-name & {:as inner-query-clauses}]
+  {:name          card-name
+   :database_id   (data/id)
+   :dataset_query {:database (data/id)
+                   :type     :query
+                   :query    inner-query-clauses}})
+
+(defn- saved-questions-virtual-db {:style/indent 0} [& card-tables]
+  {:name               "Saved Questions"
+   :id                 database/virtual-id
+   :features           ["basic-aggregations"]
+   :tables             card-tables
+   :is_saved_questions true})
+
+(defn- virtual-table-for-card [card & {:as kvs}]
+  (merge {:id           (format "card__%d" (u/get-id card))
+          :db_id        database/virtual-id
+          :display_name (:name card)
+          :schema       "All questions"
+          :description  nil}
+         kvs))
+
+(tt/expect-with-temp [Card [card (card-with-native-query "Kanye West Quote Views Per Month")]]
+  (saved-questions-virtual-db
+    (virtual-table-for-card card))
+  (do
+    ;; run the Card which will populate its result_metadata column
+    ((user->client :crowberto) :post 200 (format "card/%d/query" (u/get-id card)))
+    ;; Now fetch the database list. The 'Saved Questions' DB should be last on the list
+    (last ((user->client :crowberto) :get 200 "database" :include_cards true))))
+
+;; make sure that GET /api/database?include_cards=true groups pretends COLLECTIONS are SCHEMAS
+(tt/expect-with-temp [Collection [stamp-collection {:name "Stamps"}]
+                      Collection [coin-collection  {:name "Coins"}]
+                      Card       [stamp-card (card-with-native-query "Total Stamp Count", :collection_id (u/get-id stamp-collection))]
+                      Card       [coin-card  (card-with-native-query "Total Coin Count",  :collection_id (u/get-id coin-collection))]]
+  (saved-questions-virtual-db
+    (virtual-table-for-card coin-card  :schema "Coins")
+    (virtual-table-for-card stamp-card :schema "Stamps"))
+  (do
+    ;; run the Cards which will populate their result_metadata columns
+    (doseq [card [stamp-card coin-card]]
+      ((user->client :crowberto) :post 200 (format "card/%d/query" (u/get-id card))))
+    ;; Now fetch the database list. The 'Saved Questions' DB should be last on the list. Cards should have their Collection name as their Schema
+    (last ((user->client :crowberto) :get 200 "database" :include_cards true))))
+
+(defn- fetch-virtual-database []
+  (some #(when (= (:name %) "Saved Questions")
+           %)
+        ((user->client :crowberto) :get 200 "database" :include_cards true)))
+
+;; make sure that GET /api/database?include_cards=true removes Cards that have ambiguous columns
+(tt/expect-with-temp [Card [ok-card         (assoc (card-with-native-query "OK Card")         :result_metadata [{:name "cam"}])]
+                      Card [cambiguous-card (assoc (card-with-native-query "Cambiguous Card") :result_metadata [{:name "cam"} {:name "cam_2"}])]]
+  (saved-questions-virtual-db
+    (virtual-table-for-card ok-card))
+  (fetch-virtual-database))
+
+
+;; make sure that GET /api/database?include_cards=true removes Cards that use cumulative-sum and cumulative-count aggregations
+(defn- ok-mbql-card []
+  (assoc (card-with-mbql-query "OK Card"
+           :source-table (data/id :checkins))
+    :result_metadata [{:name "num_toucans"}]))
+
+;; cum count using the new-style multiple aggregation syntax
+(tt/expect-with-temp [Card [ok-card (ok-mbql-card)]
+                      Card [_ (assoc (card-with-mbql-query "Cum Count Card"
+                                       :source-table (data/id :checkins)
+                                       :aggregation  [[:cum-count]]
+                                       :breakout     [[:datetime-field [:field-id (data/id :checkins :date) :month]]])
+                                :result_metadata [{:name "num_toucans"}])]]
+  (saved-questions-virtual-db
+    (virtual-table-for-card ok-card))
+  (fetch-virtual-database))
+
+;; cum sum using old-style single aggregation syntax
+(tt/expect-with-temp [Card [ok-card (ok-mbql-card)]
+                      Card [_ (assoc (card-with-mbql-query "Cum Sum Card"
+                                       :source-table (data/id :checkins)
+                                       :aggregation  [:cum-sum]
+                                       :breakout     [[:datetime-field [:field-id (data/id :checkins :date) :month]]])
+                                :result_metadata [{:name "num_toucans"}])]]
+  (saved-questions-virtual-db
+    (virtual-table-for-card ok-card))
+  (fetch-virtual-database))
+
+
+;; make sure that GET /api/database/:id/metadata works for the Saved Questions 'virtual' database
+(tt/expect-with-temp [Card [card (assoc (card-with-native-query "Birthday Card") :result_metadata [{:name "age_in_bird_years"}])]]
+  (saved-questions-virtual-db
+    (assoc (virtual-table-for-card card)
+      :fields [{:name         "age_in_bird_years"
+                :table_id     (str "card__" (u/get-id card))
+                :id           ["field-literal" "age_in_bird_years" "type/*"]
+                :special_type nil}]))
+  ((user->client :crowberto) :get 200 (format "database/%d/metadata" database/virtual-id)))
+
+;; if no eligible Saved Questions exist the virtual DB metadata endpoint should just return `nil`
+(expect
+  nil
+  ((user->client :crowberto) :get 200 (format "database/%d/metadata" database/virtual-id)))
diff --git a/test/metabase/api/dataset_test.clj b/test/metabase/api/dataset_test.clj
index 1355b9394aa9a781cdfac6a53f39c14492fb5d11..ca0ab8aef463964486c23b0662740c1253c9808a 100644
--- a/test/metabase/api/dataset_test.clj
+++ b/test/metabase/api/dataset_test.clj
@@ -1,6 +1,7 @@
 (ns metabase.api.dataset-test
   "Unit tests for /api/dataset endpoints."
   (:require [expectations :refer :all]
+            [medley.core :as m]
             [metabase.api.dataset :refer [default-query-constraints]]
             [metabase.models.query-execution :refer [QueryExecution]]
             [metabase.query-processor.expand :as ql]
@@ -36,7 +37,7 @@
                    [k (f v)]))))))
 
 (defn format-response [m]
-  (into {} (for [[k v] m]
+  (into {} (for [[k v] (m/dissoc-in m [:data :results_metadata])]
              (cond
                (contains? #{:id :started_at :running_time :hash} k) [k (boolean v)]
                (= :data k) [k (if-not (contains? v :native_form)
diff --git a/test/metabase/api/field_test.clj b/test/metabase/api/field_test.clj
index 6564bdde4dd62210fec1b269cad38bc68417d356..fcebc07ac1bdf80c2ed1dd492a3197704c5965fc 100644
--- a/test/metabase/api/field_test.clj
+++ b/test/metabase/api/field_test.clj
@@ -9,6 +9,7 @@
              [data :refer :all]
              [util :as tu]]
             [metabase.test.data.users :refer :all]
+            [ring.util.codec :as codec]
             [toucan.db :as db]
             [toucan.util.test :as tt]))
 
@@ -170,6 +171,13 @@
   ((user->client :rasta) :get 200 (format "field/%d/values" (id :venues :id))))
 
 
+;; Check that trying to get values for a 'virtual' field just returns a blank values map
+(expect
+  {:values                {}
+   :human_readable_values {}}
+  ((user->client :rasta) :get 200 (format "field/%s/values" (codec/url-encode "field-literal,created_at,type/Datetime"))))
+
+
 ;; ## POST /api/field/:id/value_map_update
 
 ;; Check that we can set values
diff --git a/test/metabase/api/session_test.clj b/test/metabase/api/session_test.clj
index 4b8d97e722071bac826bed21aaa8dbfc9a480a51..5dda57327cc527ac28a25153eae1c2c85badc9aa 100644
--- a/test/metabase/api/session_test.clj
+++ b/test/metabase/api/session_test.clj
@@ -11,8 +11,9 @@
              [user :refer [User]]]
             [metabase.test
              [data :refer :all]
-             [util :as tu :refer [resolve-private-vars]]]
+             [util :as tu :refer [resolve-private-vars with-temporary-setting-values]]]
             [metabase.test.data.users :refer :all]
+            [metabase.test.integrations.ldap :refer [expect-with-ldap-server]]
             [toucan.db :as db]
             [toucan.util.test :as tt]))
 
@@ -24,11 +25,11 @@
       (tu/is-uuid-string? (:id (client :post 200 "session" (user->credentials :rasta))))))
 
 ;; Test for required params
-(expect {:errors {:email "value must be a valid email address."}}
+(expect {:errors {:username "value must be a non-blank string."}}
   (client :post 400 "session" {}))
 
 (expect {:errors {:password "value must be a non-blank string."}}
-  (client :post 400 "session" {:email "anything@metabase.com"}))
+  (client :post 400 "session" {:username "anything@metabase.com"}))
 
 ;; Test for inactive user (user shouldn't be able to login if :is_active = false)
 ;; Return same error as incorrect password to avoid leaking existence of user
@@ -43,9 +44,9 @@
 ;; Test that people get blocked from attempting to login if they try too many times
 ;; (Check that throttling works at the API level -- more tests in the throttle library itself: https://github.com/metabase/throttle)
 (expect
-    [{:errors {:email "Too many attempts! You must wait 15 seconds before trying again."}}
-     {:errors {:email "Too many attempts! You must wait 15 seconds before trying again."}}]
-  (let [login #(client :post 400 "session" {:email "fakeaccount3000@metabase.com", :password "toucans"})]
+    [{:errors {:username "Too many attempts! You must wait 15 seconds before trying again."}}
+     {:errors {:username "Too many attempts! You must wait 15 seconds before trying again."}}]
+  (let [login #(client :post 400 "session" {:username "fakeaccount3000@metabase.com", :password "toucans"})]
     ;; attempt to log in 10 times
     (dorun (repeatedly 10 login))
     ;; throttling should now be triggered
@@ -74,7 +75,7 @@
     (db/update! User (user->id :rasta), :reset_token nil, :reset_triggered nil)
     (assert (not (reset-fields-set?)))
     ;; issue reset request (token & timestamp should be saved)
-    ((user->client :rasta) :post 200 "session/forgot_password" {:email (:email (user->credentials :rasta))})
+    ((user->client :rasta) :post 200 "session/forgot_password" {:email (:username (user->credentials :rasta))})
     ;; TODO - how can we test email sent here?
     (reset-fields-set?)))
 
@@ -99,9 +100,9 @@
       (let [token (u/prog1 (str id "_" (java.util.UUID/randomUUID))
                     (db/update! User id, :reset_token <>))
             creds {:old {:password (:old password)
-                         :email    email}
+                         :username email}
                    :new {:password (:new password)
-                         :email    email}}]
+                         :username email}}]
         ;; Check that creds work
         (client :post 200 "session" (:old creds))
 
@@ -251,3 +252,40 @@
                                      admin-email                             "rasta@toucans.com"]
     (u/prog1 (is-session? (google-auth-fetch-or-create-user! "Rasta" "Toucan" "rasta@sf-toucannery.com"))
       (db/delete! User :email "rasta@sf-toucannery.com"))))                                       ; clean up after ourselves
+
+
+;;; ------------------------------------------------------------ TESTS FOR LDAP AUTH STUFF ------------------------------------------------------------
+
+;; Test that we can login with LDAP
+(expect-with-ldap-server
+  true
+  ;; delete all other sessions for the bird first, otherwise test doesn't seem to work (TODO - why?)
+  (do (db/simple-delete! Session, :user_id (user->id :rasta))
+      (tu/is-uuid-string? (:id (client :post 200 "session" (user->credentials :rasta))))))
+
+;; Test that login will fallback to local for users not in LDAP
+(expect-with-ldap-server
+  true
+  ;; delete all other sessions for the bird first, otherwise test doesn't seem to work (TODO - why?)
+  (do (db/simple-delete! Session, :user_id (user->id :crowberto))
+      (tu/is-uuid-string? (:id (client :post 200 "session" (user->credentials :crowberto))))))
+
+;; Test that login will NOT fallback for users in LDAP but with an invalid password
+(expect-with-ldap-server
+  {:errors {:password "did not match stored password"}}
+  (client :post 400 "session" (user->credentials :lucky))) ; NOTE: there's a different password in LDAP for Lucky
+
+;; Test that login will fallback to local for broken LDAP settings
+;; NOTE: This will ERROR out in the logs, it's normal
+(expect-with-ldap-server
+  true
+  (tu/with-temporary-setting-values [ldap-user-base "cn=wrong,cn=com"]
+    ;; delete all other sessions for the bird first, otherwise test doesn't seem to work (TODO - why?)
+    (do (db/simple-delete! Session, :user_id (user->id :rasta))
+        (tu/is-uuid-string? (:id (client :post 200 "session" (user->credentials :rasta)))))))
+
+;; Test that we can login with LDAP with new user
+(expect-with-ldap-server
+  true
+  (u/prog1 (tu/is-uuid-string? (:id (client :post 200 "session" {:username "sbrown20", :password "1234"})))
+    (db/delete! User :email "sally.brown@metabase.com"))) ; clean up
diff --git a/test/metabase/api/table_test.clj b/test/metabase/api/table_test.clj
index b01c49d8c60f2ba490a6ead7ae00857ff5dff1ea..6472b51930138b6cad8aa7f029f90b1f7cf77782 100644
--- a/test/metabase/api/table_test.clj
+++ b/test/metabase/api/table_test.clj
@@ -2,6 +2,7 @@
   "Tests for /api/table endpoints."
   (:require [clojure.walk :as walk]
             [expectations :refer :all]
+            [medley.core :as m]
             [metabase
              [driver :as driver]
              [http-client :as http]
@@ -9,13 +10,14 @@
              [sync-database :as sync-database]
              [util :as u]]
             [metabase.models
-             [database :refer [Database]]
+             [card :refer [Card]]
+             [database :as database :refer [Database]]
              [field :refer [Field]]
              [permissions :as perms]
              [permissions-group :as perms-group]
              [table :refer [Table]]]
             [metabase.test
-             [data :refer :all]
+             [data :as data :refer :all]
              [util :as tu :refer [match-$ resolve-private-vars]]]
             [metabase.test.data
              [dataset-definitions :as defs]
@@ -23,8 +25,7 @@
             [toucan
              [db :as db]
              [hydrate :as hydrate]]
-            [toucan.util.test :as tt]
-            [medley.core :as m]))
+            [toucan.util.test :as tt]))
 
 (resolve-private-vars metabase.models.table pk-field-id)
 (resolve-private-vars metabase.api.table dimension-options-for-response datetime-dimension-indexes numeric-dimension-indexes)
@@ -488,6 +489,40 @@
                                                               :created_at   $}))}))}])
   ((user->client :rasta) :get 200 (format "table/%d/fks" (id :users))))
 
+;; Make sure metadata for 'virtual' tables comes back as expected from GET /api/table/:id/query_metadata
+(tt/expect-with-temp [Card [card {:name          "Go Dubs!"
+                                  :database_id   (data/id)
+                                  :dataset_query {:database (data/id)
+                                                  :type     :native
+                                                  :native   {:query (format "SELECT NAME, ID, PRICE, LATITUDE FROM VENUES")}}}]]
+  (let [card-virtual-table-id (str "card__" (u/get-id card))]
+    {:display_name "Go Dubs!"
+     :schema       "All questions"
+     :db_id        database/virtual-id
+     :id           card-virtual-table-id
+     :description  nil
+     :fields       (for [[field-name display-name base-type] [["NAME"     "Name"     "type/Text"]
+                                                              ["ID"       "ID"       "type/Integer"]
+                                                              ["PRICE"    "Price"    "type/Integer"]
+                                                              ["LATITUDE" "Latitude" "type/Float"]]]
+                     {:name         field-name
+                      :display_name display-name
+                      :base_type    base-type
+                      :table_id     card-virtual-table-id
+                      :id           ["field-literal" field-name base-type]
+                      :special_type nil})})
+  (do
+    ;; run the Card which will populate its result_metadata column
+    ((user->client :crowberto) :post 200 (format "card/%d/query" (u/get-id card)))
+    ;; Now fetch the metadata for this "table"
+    ((user->client :crowberto) :get 200 (format "table/card__%d/query_metadata" (u/get-id card)))))
+
+
+;; make sure GET /api/table/:id/fks just returns nothing for 'virtual' tables
+(expect
+  []
+  ((user->client :crowberto) :get 200 "table/card__1000/fks"))
+
 ;; Ensure dimensions options are sorted numerically, but returned as strings
 (expect
   (map str (sort (map #(Long/parseLong %) (var-get datetime-dimension-indexes))))
@@ -524,13 +559,17 @@
 
 ;; Lat/Long fields should use bin-width rather than num-bins
 (expect
-  #{"bin-width" "default"}
+  (if (binning-supported?)
+    #{"bin-width" "default"}
+    #{})
   (let [response ((user->client :rasta) :get 200 (format "table/%d/query_metadata" (id :venues)))]
     (extract-dimension-options response "LATITUDE")))
 
 ;; Number columns without a special type should use "num-bins"
 (expect
-  #{"num-bins" "default"}
+  (if (binning-supported?)
+    #{"num-bins" "default"}
+    #{})
   (let [{:keys [special_type]} (Field (id :venues :price))]
     (try
       (db/update! Field (id :venues :price) :special_type nil)
diff --git a/test/metabase/api/user_test.clj b/test/metabase/api/user_test.clj
index 015c5d0884d08d08c8464f7e197fbd9f0c41601d..8e82e371e6890feb60e58ffeb2798ab0fe10a31c 100644
--- a/test/metabase/api/user_test.clj
+++ b/test/metabase/api/user_test.clj
@@ -32,7 +32,8 @@
        :last_login   $
        :first_name   "Crowberto"
        :email        "crowberto@metabase.com"
-       :google_auth  false})
+       :google_auth  false
+       :ldap_auth    false})
     (match-$ (fetch-user :lucky)
       {:common_name  "Lucky Pigeon"
        :last_name    "Pigeon"
@@ -41,7 +42,8 @@
        :last_login   $
        :first_name   "Lucky"
        :email        "lucky@metabase.com"
-       :google_auth  false})
+       :google_auth  false
+       :ldap_auth    false})
     (match-$ (fetch-user :rasta)
       {:common_name  "Rasta Toucan"
        :last_name    "Toucan"
@@ -50,7 +52,8 @@
        :last_login   $
        :first_name   "Rasta"
        :email        "rasta@metabase.com"
-       :google_auth  false})}
+       :google_auth  false
+       :ldap_auth    false})}
   (do
     ;; Delete all the other random Users we've created so far
     (let [user-ids (set (map user->id [:crowberto :rasta :lucky :trashbird]))]
@@ -132,6 +135,7 @@
      :is_superuser false
      :is_qbnewb    true
      :google_auth  false
+     :ldap_auth    false
      :id           $})
   ((user->client :rasta) :get 200 "user/current"))
 
@@ -209,7 +213,7 @@
 ;; Test that a User can change their password (superuser and non-superuser)
 (defn- user-can-reset-password? [superuser?]
   (tt/with-temp User [user {:password "def", :is_superuser (boolean superuser?)}]
-    (let [creds           {:email (:email user), :password "def"}
+    (let [creds           {:username (:email user), :password "def"}
           hashed-password (db/select-one-field :password User, :email (:email user))]
       ;; use API to reset the users password
       (http/client creds :put 200 (format "user/%d/password" (:id user)) {:password     "abc123!!DEF"
@@ -247,7 +251,7 @@
                                  :last_name  (random-name)
                                  :email      "def@metabase.com"
                                  :password   "def123"}]
-    (let [creds {:email    "def@metabase.com"
+    (let [creds {:username "def@metabase.com"
                  :password "def123"}]
       [(metabase.http-client/client creds :put 200 (format "user/%d/qbnewb" id))
        (db/select-one-field :is_qbnewb User, :id id)])))
diff --git a/test/metabase/driver/bigquery_test.clj b/test/metabase/driver/bigquery_test.clj
index 2105db2bae7ed78a1c33deeb1ee1d1c712202b60..571c0160be03c443fffa9e179e0caef036f06b1f 100644
--- a/test/metabase/driver/bigquery_test.clj
+++ b/test/metabase/driver/bigquery_test.clj
@@ -21,10 +21,10 @@
 
 ;; make sure that BigQuery native queries maintain the column ordering specified in the SQL -- post-processing ordering shouldn't apply (Issue #2821)
 (expect-with-engine :bigquery
-  {:columns ["venue_id" "user_id" "checkins_id"]
-   :cols    [{:name "venue_id",    :base_type :type/Integer}
-             {:name "user_id",     :base_type :type/Integer}
-             {:name "checkins_id", :base_type :type/Integer}]}
+  {:cols    [{:name "venue_id",    :display_name "Venue ID",    :base_type :type/Integer}
+             {:name "user_id",     :display_name  "User ID"      :base_type :type/Integer}
+             {:name "checkins_id", :display_name "Checkins ID"  :base_type :type/Integer}],
+   :columns ["venue_id" "user_id" "checkins_id"]}
   (select-keys (:data (qp/process-query {:native   {:query "SELECT [test_data.checkins.venue_id] AS [venue_id], [test_data.checkins.user_id] AS [user_id], [test_data.checkins.id] AS [checkins_id]
                                                             FROM [test_data.checkins]
                                                             LIMIT 2"}
diff --git a/test/metabase/driver/druid_test.clj b/test/metabase/driver/druid_test.clj
index 224150164f2a4c02a627f67553c382dc6ff73844..436ba38d1f8217607468d1eb6e4febb2bb518a8c 100644
--- a/test/metabase/driver/druid_test.clj
+++ b/test/metabase/driver/druid_test.clj
@@ -1,6 +1,7 @@
 (ns metabase.driver.druid-test
   (:require [cheshire.core :as json]
             [expectations :refer [expect]]
+            [medley.core :as m]
             [metabase
              [driver :as driver]
              [query-processor :as qp]
@@ -29,9 +30,10 @@
 (defn- process-native-query [query]
   (datasets/with-engine :druid
     (timeseries-qp-test/with-flattened-dbdef
-      (qp/process-query {:native   {:query query}
-                         :type     :native
-                         :database (data/id)}))))
+      (-> (qp/process-query {:native   {:query query}
+                             :type     :native
+                             :database (data/id)})
+          (m/dissoc-in [:data :results_metadata])))))
 
 ;; test druid native queries
 (expect-with-engine :druid
diff --git a/test/metabase/driver/generic_sql/native_test.clj b/test/metabase/driver/generic_sql/native_test.clj
index 1f9445e9cfba364c0d47c3c554ddf74cfb26b833..f466b820b2fd3cdc8544ecd2f306c47b607a9d7c 100644
--- a/test/metabase/driver/generic_sql/native_test.clj
+++ b/test/metabase/driver/generic_sql/native_test.clj
@@ -15,9 +15,10 @@
                :columns     ["ID"]
                :cols        [{:name "ID", :display_name "ID", :base_type :type/Integer}]
                :native_form {:query "SELECT ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}}}
-  (qp/process-query {:native   {:query "SELECT ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}
-                     :type     :native
-                     :database (id)}))
+  (-> (qp/process-query {:native   {:query "SELECT ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}
+                         :type     :native
+                         :database (id)})
+      (m/dissoc-in [:data :results_metadata])))
 
 ;; Check that column ordering is maintained
 (expect
@@ -30,9 +31,10 @@
                              {:name "NAME",        :display_name "Name",        :base_type :type/Text}
                              {:name "CATEGORY_ID", :display_name "Category ID", :base_type :type/Integer}]
                :native_form {:query "SELECT ID, NAME, CATEGORY_ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}}}
-  (qp/process-query {:native   {:query "SELECT ID, NAME, CATEGORY_ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}
-                     :type     :native
-                     :database (id)}))
+  (-> (qp/process-query {:native   {:query "SELECT ID, NAME, CATEGORY_ID FROM VENUES ORDER BY ID DESC LIMIT 2;"}
+                         :type     :native
+                         :database (id)})
+      (m/dissoc-in [:data :results_metadata])))
 
 ;; Check that we get proper error responses for malformed SQL
 (expect {:status :failed
diff --git a/test/metabase/driver/mongo_test.clj b/test/metabase/driver/mongo_test.clj
index 102e57808d429e4c6a6adda72cefb6b923af9610..edbce9b4d472c202a26e1a9f123b477a35f8e6b1 100644
--- a/test/metabase/driver/mongo_test.clj
+++ b/test/metabase/driver/mongo_test.clj
@@ -1,6 +1,7 @@
 (ns metabase.driver.mongo-test
   "Tests for Mongo driver."
   (:require [expectations :refer :all]
+            [medley.core :as m]
             [metabase
              [driver :as driver]
              [query-processor :as qp]
@@ -80,10 +81,11 @@
                :cols        [{:name "count", :display_name "Count", :base_type :type/Integer}]
                :native_form {:collection "venues"
                              :query      native-query}}}
-  (qp/process-query {:native   {:query      native-query
-                                :collection "venues"}
-                     :type     :native
-                     :database (data/id)}))
+  (-> (qp/process-query {:native   {:query      native-query
+                                    :collection "venues"}
+                         :type     :native
+                         :database (data/id)})
+      (m/dissoc-in [:data :results_metadata])))
 
 ;; ## Tests for individual syncing functions
 
diff --git a/test/metabase/driver/postgres_test.clj b/test/metabase/driver/postgres_test.clj
index 554e9dd363465246b5e6517bc980e6d8e86946ff..ffd9889adbe6b663b560aa3f8de1ca30ed6a65eb 100644
--- a/test/metabase/driver/postgres_test.clj
+++ b/test/metabase/driver/postgres_test.clj
@@ -120,7 +120,7 @@
              [3 "ouija_board"]]}
   (-> (data/dataset metabase.driver.postgres-test/dots-in-names
         (data/run-query objects.stuff))
-      :data (dissoc :cols :native_form)))
+      :data (dissoc :cols :native_form :results_metadata)))
 
 
 ;; Make sure that duplicate column names (e.g. caused by using a FK) still return both columns
@@ -140,7 +140,7 @@
   (-> (data/dataset metabase.driver.postgres-test/duplicate-names
         (data/run-query people
           (ql/fields $name $bird_id->birds.name)))
-      :data (dissoc :cols :native_form)))
+      :data (dissoc :cols :native_form :results_metadata)))
 
 
 ;;; Check support for `inet` columns
diff --git a/test/metabase/driver/presto_test.clj b/test/metabase/driver/presto_test.clj
index ffeca83d0e4424e0863a85f4b8c840bdaadff06f..9832c924be3b3d5bc27929bb10464476c520f58c 100644
--- a/test/metabase/driver/presto_test.clj
+++ b/test/metabase/driver/presto_test.clj
@@ -10,7 +10,7 @@
             [toucan.db :as db])
   (:import metabase.driver.presto.PrestoDriver))
 
-(resolve-private-vars metabase.driver.presto details->uri details->request parse-presto-results quote-name quote+combine-names apply-page)
+(resolve-private-vars metabase.driver.presto details->uri details->request parse-presto-results quote-name quote+combine-names rename-duplicates apply-page)
 
 ;;; HELPERS
 
@@ -53,6 +53,11 @@
   (parse-presto-results [{:type "date"} {:type "timestamp with time zone"} {:type "timestamp"} {:type "decimal(10,4)"} {:type "varchar(255)"}]
                         [["2017-04-03", "2017-04-03 10:19:17.417 America/Toronto", "2017-04-03 10:19:17.417", "3.1416", "test"]]))
 
+(expect
+  [[0, false, "", nil]]
+  (parse-presto-results [{:type "integer"} {:type "boolean"} {:type "varchar(255)"} {:type "date"}]
+                        [[0, false, "", nil]]))
+
 (expect
   "\"weird.table\"\" name\""
   (quote-name "weird.table\" name"))
@@ -61,6 +66,10 @@
   "\"weird . \"\"schema\".\"weird.table\"\" name\""
   (quote+combine-names "weird . \"schema" "weird.table\" name"))
 
+(expect
+  ["name" "count" "count_2" "sum", "sum_2", "sum_3"]
+  (rename-duplicates ["name" "count" "count" "sum" "sum" "sum"]))
+
 ;; DESCRIBE-DATABASE
 (datasets/expect-with-engine :presto
   {:tables #{{:name "categories" :schema "default"}
@@ -90,12 +99,13 @@
 ;;; ANALYZE-TABLE
 (datasets/expect-with-engine :presto
   {:row_count 100
-   :fields    [{:id (data/id :venues :category_id), :values [2 3 4 5 6 7 10 11 12 13 14 15 18 19 20 29 40 43 44 46 48 49 50 58 64 67 71 74]}
-               {:id (data/id :venues :id)}
-               {:id (data/id :venues :latitude)}
-               {:id (data/id :venues :longitude)}
+   :fields    [{:id        (data/id :venues :category_id), :values    [2 3 4 5 6 7 10 11 12 13 14 15 18 19 20 29 40 43 44 46 48 49 50 58 64 67 71 74]
+                :min-value 2.0,                            :max-value 74.0}
+               {:id (data/id :venues :id), :min-value 1.0, :max-value 100.0}
+               {:id (data/id :venues :latitude), :min-value 10.0646, :max-value 40.7794}
+               {:id (data/id :venues :longitude), :min-value -165.374, :max-value -73.9533}
                {:id (data/id :venues :name), :values (db/select-one-field :values 'FieldValues, :field_id (data/id :venues :name))}
-               {:id (data/id :venues :price), :values [1 2 3 4]}]}
+               {:id (data/id :venues :price), :values [1 2 3 4], :min-value 1.0, :max-value 4.0}]}
   (let [venues-table (db/select-one 'Table :id (data/id :venues))]
     (driver/analyze-table (PrestoDriver.) venues-table (set (mapv :id (table/fields venues-table))))))
 
diff --git a/test/metabase/events/activity_feed_test.clj b/test/metabase/events/activity_feed_test.clj
index 9f243b06e06031e809a01949631846f3f3cb669f..2ae4b9e8ebf15b16553b64045d410a7831c8f531 100644
--- a/test/metabase/events/activity_feed_test.clj
+++ b/test/metabase/events/activity_feed_test.clj
@@ -9,8 +9,9 @@
              [metric :refer [Metric]]
              [pulse :refer [Pulse]]
              [segment :refer [Segment]]]
-            [metabase.test.data :refer :all]
+            [metabase.test.data :as data :refer :all]
             [metabase.test.data.users :refer [user->id]]
+            [metabase.util :as u]
             [toucan.db :as db]
             [toucan.util.test :as tt]))
 
@@ -40,6 +41,27 @@
         :topic    "card-create"
         :model_id (:id card)))))
 
+;; when I save a Card that uses a NESTED query, is the activity recorded? :D
+(expect
+  {:topic       :card-create
+   :user_id     (user->id :rasta)
+   :model       "card"
+   :database_id (data/id)
+   :table_id    (data/id :venues)
+   :details     {:name "My Cool NESTED Card", :description nil}}
+  (tt/with-temp* [Card [card-1 {:name          "My Cool Card"
+                                :dataset_query {:database (data/id)
+                                                :type     :query
+                                                :query    {:source-table (data/id :venues)}}}]
+                  Card [card-2 {:name          "My Cool NESTED Card"
+                                :dataset_query {:database metabase.models.database/virtual-id
+                                                :type     :query
+                                                :query    {:source-table (str "card__" (u/get-id card-1))}}}]]
+    (with-temp-activities
+      (process-activity-event! {:topic :card-create, :item card-2})
+      (db/select-one [Activity :topic :user_id :model :database_id :table_id :details]
+        :topic    "card-create"
+        :model_id (:id card-2)))))
 
 
 ;; `:card-update` event
diff --git a/test/metabase/events/revision_test.clj b/test/metabase/events/revision_test.clj
index 7fbb6e7633d589194f9b39d5282a291b1fb304a2..85ce227362a4d8e699d39a4e354630c32a251dfa 100644
--- a/test/metabase/events/revision_test.clj
+++ b/test/metabase/events/revision_test.clj
@@ -44,7 +44,8 @@
    :cache_ttl              nil
    :query_type             "query"
    :table_id               (id :categories)
-   :visualization_settings {}})
+   :visualization_settings {}
+   :result_metadata        nil})
 
 (defn- dashboard->revision-object [dashboard]
   {:description  nil
diff --git a/test/metabase/http_client.clj b/test/metabase/http_client.clj
index b16790230d69800d76db42b769cca561eeb139bc..9de0ef75217a603e25d8bce9405404d62d2ae723 100644
--- a/test/metabase/http_client.clj
+++ b/test/metabase/http_client.clj
@@ -63,14 +63,14 @@
 (declare client)
 
 (defn authenticate
-  "Authenticate a test user with EMAIL and PASSWORD, returning their Metabase Session token;
+  "Authenticate a test user with USERNAME and PASSWORD, returning their Metabase Session token;
    or throw an Exception if that fails."
-  [{:keys [email password], :as credentials}]
-  {:pre [(string? email) (string? password)]}
+  [{:keys [username password], :as credentials}]
+  {:pre [(string? username) (string? password)]}
   (try
     (:id (client :post 200 "session" credentials))
     (catch Throwable e
-      (log/error "Failed to authenticate with email:" email "and password:" password ":" (.getMessage e)))))
+      (log/error "Failed to authenticate with username:" username "and password:" password ":" (.getMessage e)))))
 
 
 ;;; client
@@ -98,12 +98,11 @@
         (log/error (u/pprint-to-str 'red body))
         (throw (ex-info message {:status-code actual-status-code}))))))
 
-(defn- method->request-fn [method]
-  (case method
-    :get    client/get
-    :post   client/post
-    :put    client/put
-    :delete client/delete))
+(def ^:private method->request-fn
+  {:get    client/get
+   :post   client/post
+   :put    client/put
+   :delete client/delete})
 
 (defn- -client [credentials method expected-status url http-body url-param-kwargs request-options]
   ;; Since the params for this function can get a little complicated make sure we validate them
@@ -141,7 +140,7 @@
 
   Args:
 
-   *  CREDENTIALS           Optional map of `:email` and `:password` or `X-METABASE-SESSION` token of a User who we should perform the request as
+   *  CREDENTIALS           Optional map of `:username` and `:password` or `X-METABASE-SESSION` token of a User who we should perform the request as
    *  METHOD                `:get`, `:post`, `:delete`, or `:put`
    *  EXPECTED-STATUS-CODE  When passed, throw an exception if the response has a different status code.
    *  URL                   Base URL of the request, which will be appended to `*url-prefix*`. e.g. `card/1/favorite`
diff --git a/test/metabase/integrations/ldap_test.clj b/test/metabase/integrations/ldap_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..5e1942a83eb786cf5ee6bdff8593feb054112415
--- /dev/null
+++ b/test/metabase/integrations/ldap_test.clj
@@ -0,0 +1,101 @@
+(ns metabase.integrations.ldap-test
+  (:require [expectations :refer :all]
+            [metabase.integrations.ldap :as ldap]
+            (metabase.test [util :refer [resolve-private-vars]])
+            (metabase.test.integrations [ldap :refer [expect-with-ldap-server get-ldap-port]])))
+
+(resolve-private-vars metabase.integrations.ldap escape-value settings->ldap-options get-connection get-user-groups)
+
+
+(defn- get-ldap-details []
+  {:host       "localhost"
+   :port       (get-ldap-port)
+   :bind-dn    "cn=Directory Manager"
+   :password   "password"
+   :security   "none"
+   :user-base  "dc=metabase,dc=com"
+   :group-base "dc=metabase,dc=com"})
+
+;; See test_resources/ldap.ldif for fixtures
+
+(expect
+  "\\2AJohn \\28Dude\\29 Doe\\5C"
+  (escape-value "*John (Dude) Doe\\"))
+
+;; The connection test should pass with valid settings
+(expect-with-ldap-server
+  {:status :SUCCESS}
+  (ldap/test-ldap-connection (get-ldap-details)))
+
+;; The connection test should allow anonymous binds
+(expect-with-ldap-server
+  {:status :SUCCESS}
+  (ldap/test-ldap-connection (dissoc (get-ldap-details) :bind-dn)))
+
+;; The connection test should fail with an invalid user search base
+(expect-with-ldap-server
+  :ERROR
+  (:status (ldap/test-ldap-connection (assoc (get-ldap-details) :user-base "dc=example,dc=com"))))
+
+;; The connection test should fail with an invalid group search base
+(expect-with-ldap-server
+  :ERROR
+  (:status (ldap/test-ldap-connection (assoc (get-ldap-details) :group-base "dc=example,dc=com"))))
+
+;; The connection test should fail with an invalid bind DN
+(expect-with-ldap-server
+  :ERROR
+  (:status (ldap/test-ldap-connection (assoc (get-ldap-details) :bind-dn "cn=Not Directory Manager"))))
+
+;; The connection test should fail with an invalid bind password
+(expect-with-ldap-server
+  :ERROR
+  (:status (ldap/test-ldap-connection (assoc (get-ldap-details) :password "wrong"))))
+
+;; Make sure the basic connection stuff works, this will throw otherwise
+(expect-with-ldap-server
+  nil
+  (.close (get-connection)))
+
+;; Login with everything right should succeed
+(expect-with-ldap-server
+  true
+  (ldap/verify-password "cn=Directory Manager" "password"))
+
+;; Login with wrong password should fail
+(expect-with-ldap-server
+  false
+  (ldap/verify-password "cn=Directory Manager" "wrongpassword"))
+
+;; Login with invalid DN should fail
+(expect-with-ldap-server
+  false
+  (ldap/verify-password "cn=Nobody,ou=nowhere,dc=metabase,dc=com" "password"))
+
+;; Login for regular users should also work
+(expect-with-ldap-server
+  true
+  (ldap/verify-password "cn=Sally Brown,ou=People,dc=metabase,dc=com" "1234"))
+
+;; Login for regular users should also fail for the wrong password
+(expect-with-ldap-server
+  false
+  (ldap/verify-password "cn=Sally Brown,ou=People,dc=metabase,dc=com" "password"))
+
+;; Find by username should work (given the default LDAP filter and test fixtures)
+(expect-with-ldap-server
+  {:dn         "cn=John Smith,ou=People,dc=metabase,dc=com"
+   :first-name "John"
+   :last-name  "Smith"
+   :email      "John.Smith@metabase.com"
+   :groups     ["cn=Accounting,ou=Groups,dc=metabase,dc=com"]}
+  (ldap/find-user "jsmith1"))
+
+;; Find by email should also work (also given our default settings and fixtures)
+(expect-with-ldap-server
+  {:dn         "cn=John Smith,ou=People,dc=metabase,dc=com"
+   :first-name "John"
+   :last-name  "Smith"
+   :email      "John.Smith@metabase.com"
+   :groups     ["cn=Accounting,ou=Groups,dc=metabase,dc=com"]}
+  (ldap/find-user "John.Smith@metabase.com"))
diff --git a/test/metabase/middleware_test.clj b/test/metabase/middleware_test.clj
index 90175af670f6a37eb34c3cf14472ce4a174e0676..94cf4d396ace2e23988ffac8943f6adae2d92ae7 100644
--- a/test/metabase/middleware_test.clj
+++ b/test/metabase/middleware_test.clj
@@ -15,7 +15,8 @@
             [metabase.test.data.users :refer :all]
             [ring.mock.request :as mock]
             [ring.util.response :as resp]
-            [toucan.db :as db]))
+            [toucan.db :as db]
+            [clojure.string :as string]))
 
 ;;  ===========================  TEST wrap-session-id middleware  ===========================
 
@@ -217,12 +218,12 @@
               _ (Thread/sleep 1000)
               second-second (async/poll! connection)
               eventually (apply str (async/<!! (async/into [] connection)))]
-          [first-second second-second eventually])))))
+          [first-second second-second (string/trim eventually)])))))
 
 
 ;;slow success
 (expect
-  [\newline \newline "\n\n\n{\"success\":true}"]
+  [\newline \newline "{\"success\":true}"]
   (test-streaming-endpoint streaming-slow-success))
 
 ;; immediate success should have no padding
@@ -232,7 +233,7 @@
 
 ;; we know delayed failures (exception thrown) will just drop the connection
 (expect
-  [\newline \newline "\n\n\n"]
+  [\newline \newline ""]
   (test-streaming-endpoint streaming-slow-failure))
 
 ;; immediate failures (where an exception is thown will return a 500
diff --git a/test/metabase/models/card_test.clj b/test/metabase/models/card_test.clj
index 0d7fc226e149a9d3122a0b6293c5e14053d1a114..9dde412f80c1a58365eac44e3a000ba5b47441b3 100644
--- a/test/metabase/models/card_test.clj
+++ b/test/metabase/models/card_test.clj
@@ -5,6 +5,7 @@
              [card :refer :all]
              [dashboard :refer [Dashboard]]
              [dashboard-card :refer [DashboardCard]]
+             [database :as database]
              [interface :as mi]
              [permissions :as perms]]
             [metabase.query-processor.expand :as ql]
@@ -120,6 +121,42 @@
                            (ql/order-by (ql/asc (ql/fk-> (data/id :checkins :venue_id) (data/id :venues :name))))))
                    :read))
 
+;; MBQL w/ nested MBQL query
+(defn- query-with-source-card [card]
+  {:database database/virtual-id, :type "query", :query {:source_table (str "card__" (u/get-id card))}})
+
+(expect
+  #{(perms/object-path (data/id) "PUBLIC" (data/id :venues))}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :query
+                                            :query    {:source-table (data/id :venues)}}}]
+    (query-perms-set (query-with-source-card card) :read)))
+
+;; MBQL w/ nested MBQL query including a JOIN
+(expect
+  #{(perms/object-path (data/id) "PUBLIC" (data/id :checkins))
+    (perms/object-path (data/id) "PUBLIC" (data/id :users))}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :query
+                                            :query    {:source-table (data/id :checkins)
+                                                       :order-by     [[:asc [:fk-> (data/id :checkins :user_id) (data/id :users :id)]]]}}}]
+    (query-perms-set (query-with-source-card card) :read)))
+
+;; MBQL w/ nested NATIVE query
+(expect
+  #{(perms/native-read-path (data/id))}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :native
+                                            :native   {:query "SELECT * FROM CHECKINS"}}}]
+    (query-perms-set (query-with-source-card card) :read)))
+
+(expect
+  #{(perms/native-readwrite-path (data/id))}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :native
+                                            :native   {:query "SELECT * FROM CHECKINS"}}}]
+    (query-perms-set (query-with-source-card card) :write)))
+
 ;; invalid/legacy card should return perms for something that doesn't exist so no one gets to see it
 (expect
   #{"/db/0/"}
diff --git a/test/metabase/models/pulse_test.clj b/test/metabase/models/pulse_test.clj
index 6771d417dfe97548101ba54f3373d2230c034e13..a080de15b18ba802d0474e167b3f1d9ec4eb8486 100644
--- a/test/metabase/models/pulse_test.clj
+++ b/test/metabase/models/pulse_test.clj
@@ -183,3 +183,13 @@
                                                   :recipients    [{:email "foo@bar.com"}
                                                                   {:id (user->id :crowberto)}]}]
                                 :skip-if-empty? false})))
+
+;; make sure fetching a Pulse doesn't return any archived cards
+(expect
+  1
+  (tt/with-temp* [Pulse     [pulse]
+                  Card      [card-1 {:archived true}]
+                  Card      [card-2]
+                  PulseCard [_ {:pulse_id (u/get-id pulse), :card_id (u/get-id card-1), :position 0}]
+                  PulseCard [_ {:pulse_id (u/get-id pulse), :card_id (u/get-id card-2), :position 1}]]
+    (count (:cards (retrieve-pulse (u/get-id pulse))))))
diff --git a/test/metabase/query_processor/expand_resolve_test.clj b/test/metabase/query_processor/expand_resolve_test.clj
index 06130f4f55f44e68b96c2d04ce4f534f1fa65273..0b4644356cad68dc8df68dff47962fca1c9632f6 100644
--- a/test/metabase/query_processor/expand_resolve_test.clj
+++ b/test/metabase/query_processor/expand_resolve_test.clj
@@ -23,6 +23,12 @@
   {:fk-field-id      nil, :datetime-unit nil,
    :binning-strategy nil, :binning-param nil})
 
+(def ^:private field-defaults
+  {:visibility-type :normal, :fk-field-id nil
+   :position        nil,     :description nil
+   :parent-id       nil,     :parent      nil
+   :min-value       nil,     :max-value   nil})
+
 ;; basic rows query w/ filter
 (expect
   [ ;; expanded form
@@ -42,39 +48,29 @@
                                   :name   "VENUES"
                                   :id     (id :venues)}
                    :filter       {:filter-type :>
-                                  :field       {:field-id           (id :venues :price)
-                                                :fk-field-id        nil
-                                                :field-name         "PRICE"
-                                                :field-display-name "Price"
-                                                :base-type          :type/Integer
-                                                :special-type       :type/Category
-                                                :visibility-type    :normal
-                                                :table-id           (id :venues)
-                                                :schema-name        "PUBLIC"
-                                                :table-name         "VENUES"
-                                                :position           nil
-                                                :description        nil
-                                                :parent-id          nil
-                                                :parent             nil
-                                                :min-value          1.0
-                                                :max-value          4.0}
+                                  :field       (merge field-defaults
+                                                      {:field-id           (id :venues :price)
+                                                       :field-name         "PRICE"
+                                                       :field-display-name "Price"
+                                                       :base-type          :type/Integer
+                                                       :special-type       :type/Category
+                                                       :table-id           (id :venues)
+                                                       :schema-name        "PUBLIC"
+                                                       :table-name         "VENUES"
+                                                       :min-value          1.0
+                                                       :max-value          4.0})
                                   :value       {:value 1
-                                                :field {:field-id           (id :venues :price)
-                                                        :fk-field-id        nil
-                                                        :field-name         "PRICE"
-                                                        :field-display-name "Price"
-                                                        :base-type          :type/Integer
-                                                        :special-type       :type/Category
-                                                        :visibility-type    :normal
-                                                        :table-id           (id :venues)
-                                                        :schema-name        "PUBLIC"
-                                                        :table-name         "VENUES"
-                                                        :position           nil
-                                                        :description        nil
-                                                        :parent-id          nil
-                                                        :parent             nil
-                                                        :min-value          1.0
-                                                        :max-value          4.0}}}
+                                                :field (merge field-defaults
+                                                              {:field-id           (id :venues :price)
+                                                               :field-name         "PRICE"
+                                                               :field-display-name "Price"
+                                                               :base-type          :type/Integer
+                                                               :special-type       :type/Category
+                                                               :table-id           (id :venues)
+                                                               :schema-name        "PUBLIC"
+                                                               :table-name         "VENUES"
+                                                               :min-value          1.0
+                                                               :max-value          4.0})}}
                    :join-tables  nil}
     :fk-field-ids #{}
     :table-ids    #{(id :venues)}}]
@@ -105,39 +101,27 @@
                                   :name   "VENUES"
                                   :id     (id :venues)}
                    :filter       {:filter-type :=
-                                  :field       {:field-id           (id :categories :name)
-                                                :fk-field-id        (id :venues :category_id)
-                                                :field-name         "NAME"
-                                                :field-display-name "Name"
-                                                :base-type          :type/Text
-                                                :special-type       :type/Name
-                                                :visibility-type    :normal
-                                                :table-id           (id :categories)
-                                                :schema-name        nil
-                                                :table-name         "CATEGORIES__via__CATEGORY_ID"
-                                                :position           nil
-                                                :description        nil
-                                                :parent-id          nil
-                                                :parent             nil
-                                                :min-value          nil
-                                                :max-value          nil}
+                                  :field       (merge field-defaults
+                                                      {:field-id           (id :categories :name)
+                                                       :fk-field-id        (id :venues :category_id)
+                                                       :field-name         "NAME"
+                                                       :field-display-name "Name"
+                                                       :base-type          :type/Text
+                                                       :special-type       :type/Name
+                                                       :table-id           (id :categories)
+                                                       :schema-name        nil
+                                                       :table-name         "CATEGORIES__via__CATEGORY_ID"})
                                   :value       {:value "abc"
-                                                :field {:field-id           (id :categories :name)
-                                                        :fk-field-id        (id :venues :category_id)
-                                                        :field-name         "NAME"
-                                                        :field-display-name "Name"
-                                                        :base-type          :type/Text
-                                                        :special-type       :type/Name
-                                                        :visibility-type    :normal
-                                                        :table-id           (id :categories)
-                                                        :schema-name        nil
-                                                        :table-name         "CATEGORIES__via__CATEGORY_ID"
-                                                        :position           nil
-                                                        :description        nil
-                                                        :parent-id          nil
-                                                        :parent             nil
-                                                        :min-value          nil
-                                                        :max-value          nil}}}
+                                                :field (merge field-defaults
+                                                              {:field-id           (id :categories :name)
+                                                               :fk-field-id        (id :venues :category_id)
+                                                               :field-name         "NAME"
+                                                               :field-display-name "Name"
+                                                               :base-type          :type/Text
+                                                               :special-type       :type/Name
+                                                               :table-id           (id :categories)
+                                                               :schema-name        nil
+                                                               :table-name         "CATEGORIES__via__CATEGORY_ID"})}}
                    :join-tables  [{:source-field {:field-id   (id :venues :category_id)
                                                   :field-name "CATEGORY_ID"}
                                    :pk-field     {:field-id   (id :categories :id)
@@ -149,8 +133,8 @@
     :fk-field-ids #{(id :venues :category_id)}
     :table-ids    #{(id :categories)}}]
   (let [expanded-form (ql/expand (wrap-inner-query (query venues
-                                                          (ql/filter (ql/= $category_id->categories.name
-                                                                           "abc")))))]
+                                                     (ql/filter (ql/= $category_id->categories.name
+                                                                      "abc")))))]
     (mapv obj->map [expanded-form
                     (resolve/resolve expanded-form)])))
 
@@ -178,40 +162,28 @@
                                   :name   "CHECKINS"
                                   :id     (id :checkins)}
                    :filter       {:filter-type :>
-                                  :field       {:field {:field-id           (id :users :last_login)
-                                                        :fk-field-id        (id :checkins :user_id)
-                                                        :field-name         "LAST_LOGIN"
-                                                        :field-display-name "Last Login"
-                                                        :base-type          :type/DateTime
-                                                        :special-type       nil
-                                                        :visibility-type    :normal
-                                                        :table-id           (id :users)
-                                                        :schema-name        nil
-                                                        :table-name         "USERS__via__USER_ID"
-                                                        :position           nil
-                                                        :description        nil
-                                                        :parent-id          nil
-                                                        :parent             nil
-                                                        :min-value          nil
-                                                        :max-value          nil}
+                                  :field       {:field (merge field-defaults
+                                                              {:field-id           (id :users :last_login)
+                                                               :fk-field-id        (id :checkins :user_id)
+                                                               :field-name         "LAST_LOGIN"
+                                                               :field-display-name "Last Login"
+                                                               :base-type          :type/DateTime
+                                                               :special-type       nil
+                                                               :table-id           (id :users)
+                                                               :schema-name        nil
+                                                               :table-name         "USERS__via__USER_ID"})
                                                 :unit  :year}
                                   :value       {:value (u/->Timestamp "1980-01-01")
-                                                :field {:field {:field-id           (id :users :last_login)
-                                                                :fk-field-id        (id :checkins :user_id)
-                                                                :field-name         "LAST_LOGIN"
-                                                                :field-display-name "Last Login"
-                                                                :base-type          :type/DateTime
-                                                                :special-type       nil
-                                                                :visibility-type    :normal
-                                                                :table-id           (id :users)
-                                                                :schema-name        nil
-                                                                :table-name         "USERS__via__USER_ID"
-                                                                :position           nil
-                                                                :description        nil
-                                                                :parent-id          nil
-                                                                :parent             nil
-                                                                :min-value          nil
-                                                                :max-value          nil}
+                                                :field {:field (merge field-defaults
+                                                                      {:field-id           (id :users :last_login)
+                                                                       :fk-field-id        (id :checkins :user_id)
+                                                                       :field-name         "LAST_LOGIN"
+                                                                       :field-display-name "Last Login"
+                                                                       :base-type          :type/DateTime
+                                                                       :special-type       nil
+                                                                       :table-id           (id :users)
+                                                                       :schema-name        nil
+                                                                       :table-name         "USERS__via__USER_ID"})
                                                         :unit  :year}}}
                    :join-tables  [{:source-field {:field-id   (id :checkins :user_id)
                                                   :field-name "USER_ID"}
@@ -252,38 +224,28 @@
                                   :id     (id :checkins)}
                    :aggregation  [{:aggregation-type    :sum
                                    :custom-name         nil
-                                   :field               {:description        nil
-                                                         :base-type          :type/Integer
-                                                         :parent             nil
-                                                         :table-id           (id :venues)
-                                                         :special-type       :type/Category
-                                                         :field-name         "PRICE"
-                                                         :field-display-name "Price"
-                                                         :parent-id          nil
-                                                         :visibility-type    :normal
-                                                         :position           nil
-                                                         :field-id           (id :venues :price)
-                                                         :fk-field-id        (id :checkins :venue_id)
-                                                         :table-name         "VENUES__via__VENUE_ID"
-                                                         :schema-name        nil
-                                                         :min-value          1.0
-                                                         :max-value          4.0}}]
-                   :breakout     [{:field {:description        nil
-                                           :base-type          :type/Date
-                                           :parent             nil
-                                           :table-id           (id :checkins)
-                                           :special-type       nil
-                                           :field-name         "DATE"
-                                           :field-display-name "Date"
-                                           :parent-id          nil
-                                           :visibility-type    :normal
-                                           :position           nil
-                                           :field-id           (id :checkins :date)
-                                           :fk-field-id        nil
-                                           :table-name         "CHECKINS"
-                                           :schema-name        "PUBLIC"
-                                           :min-value          nil
-                                           :max-value          nil}
+                                   :field               (merge field-defaults
+                                                               {:base-type          :type/Integer
+                                                                :table-id           (id :venues)
+                                                                :special-type       :type/Category
+                                                                :field-name         "PRICE"
+                                                                :field-display-name "Price"
+                                                                :field-id           (id :venues :price)
+                                                                :fk-field-id        (id :checkins :venue_id)
+                                                                :table-name         "VENUES__via__VENUE_ID"
+                                                                :schema-name        nil
+                                                                :min-value          1.0
+                                                                :max-value          4.0})}]
+                   :breakout     [{:field (merge field-defaults
+                                                 {:description        nil
+                                                  :base-type          :type/Date
+                                                  :table-id           (id :checkins)
+                                                  :special-type       nil
+                                                  :field-name         "DATE"
+                                                  :field-display-name "Date"
+                                                  :field-id           (id :checkins :date)
+                                                  :table-name         "CHECKINS"
+                                                  :schema-name        "PUBLIC"})
                                    :unit  :day-of-week}]
                    :join-tables  [{:source-field {:field-id   (id :checkins :venue_id)
                                                   :field-name "VENUE_ID"}
diff --git a/test/metabase/query_processor/middleware/annotate_and_sort_test.clj b/test/metabase/query_processor/middleware/annotate_and_sort_test.clj
index fabdac0715d2da0e09448d95bb13918431673173..422287b29f4915ee12048befc21da4202ef61c5c 100644
--- a/test/metabase/query_processor/middleware/annotate_and_sort_test.clj
+++ b/test/metabase/query_processor/middleware/annotate_and_sort_test.clj
@@ -1,5 +1,6 @@
 (ns metabase.query-processor.middleware.annotate-and-sort-test
   (:require [expectations :refer :all]
+            metabase.query-processor.middleware.annotate-and-sort
             [metabase.test.util :as tu]))
 
 (tu/resolve-private-vars metabase.query-processor.middleware.annotate-and-sort
diff --git a/test/metabase/query_processor/middleware/annotate_test.clj b/test/metabase/query_processor/middleware/annotate_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..e847074df8ec0607eeda180161bad6222941380a
--- /dev/null
+++ b/test/metabase/query_processor/middleware/annotate_test.clj
@@ -0,0 +1,105 @@
+(ns metabase.query-processor.middleware.annotate-test
+  (:require [expectations :refer [expect]]
+            [metabase.query-processor
+             [annotate :as annotate]
+             [interface :as qpi]]
+            [metabase.util :as u]
+            [toucan.util.test :as tt]))
+
+;; make sure when using a source query the right metadata comes back so we are able to do drill-through properly
+(expect
+  [{:field-id   [:field-literal "id" :type/Integer]
+    :field-name "id"
+    :source     :fields}
+   {:field-id   [:field-literal "reciever_id" :type/Integer]
+    :field-name "reciever_id"
+    :source     :fields}
+   {:field-id   [:field-literal "sender_id" :type/Integer]
+    :field-name "sender_id"
+    :source     :fields}
+   {:field-id   [:field-literal "text" :type/Text]
+    :field-name "text"
+    :source     :fields}]
+  (map
+   (u/rpartial select-keys [:field-id :field-name :source])
+   (annotate/collect-fields
+     {:source-query {:source-table       {:schema "public", :name "messages", :id 1}
+                     :fields-is-implicit true
+                     :fields             [(qpi/map->Field
+                                           {:field-id           1
+                                            :field-name         "id"
+                                            :field-display-name "ID"
+                                            :base-type          :type/Integer
+                                            :special-type       :type/PK
+                                            :table-id           1})
+                                          (qpi/map->Field
+                                           {:field-id           2
+                                            :field-name         "reciever_id"
+                                            :field-display-name "Rec I Ever ID"
+                                            :base-type          :type/Integer
+                                            :special-type       :type/FK
+                                            :table-id           1})
+                                          (qpi/map->Field
+                                           {:field-id           3
+                                            :field-name         "sender_id"
+                                            :field-display-name "Sender ID"
+                                            :base-type          :type/Integer
+                                            :special-type       :type/FK
+                                            :table-id           1})
+                                          (qpi/map->Field
+                                           {:field-id           3
+                                            :field-name         "text"
+                                            :field-display-name "Text"
+                                            :base-type          :type/Text
+                                            :special-type       :type/Category
+                                            :table-id           1})]}})))
+
+;; make sure when doing a breakout of a nested query the right metadata comes back (fields are "collected" properly) so things like bar charts work as expected
+(expect
+  [{:field-id [:field-literal "text"        :type/Text],    :field-name "text",        :source :breakout}
+   {:field-id [:field-literal "id"          :type/Integer], :field-name "id",          :source :fields}
+   {:field-id [:field-literal "reciever_id" :type/Integer], :field-name "reciever_id", :source :fields}
+   {:field-id [:field-literal "sender_id"   :type/Integer], :field-name "sender_id",   :source :fields}
+   {:field-id [:field-literal "text"        :type/Text],    :field-name "text",        :source :fields}
+   {:field-id [:field-literal "text"        :type/Text],    :field-name "text",        :source :order-by}
+   {:field-id [:field-literal "text"        :type/Text],    :field-name "text",        :source :order-by}]
+  (map
+   (u/rpartial select-keys [:field-id :field-name :source])
+   (annotate/collect-fields
+     {:aggregation  [{:aggregation-type :count, :custom-name nil}]
+      :breakout     [(qpi/map->FieldLiteral {:field-name "text", :base-type :type/Text, :datetime-unit nil})]
+      :source-query {:source-table       {:schema "public", :name "messages", :id 1}
+                     :fields-is-implicit true
+                     :fields             [(qpi/map->Field
+                                           {:field-id     1
+                                            :field-name   "id"
+                                            :base-type    :type/Integer
+                                            :special-type :type/PK
+                                            :table-id     1})
+                                          (qpi/map->Field
+                                           {:field-id     2
+                                            :field-name   "reciever_id"
+                                            :base-type    :type/Integer
+                                            :special-type :type/FK
+                                            :table-id     1})
+                                          (qpi/map->Field
+                                           {:field-id     3
+                                            :field-name   "sender_id"
+                                            :base-type    :type/Integer
+                                            :special-type :type/FK
+                                            :table-id     1})
+                                          (qpi/map->Field
+                                           {:field-id     4
+                                            :field-name   "text"
+                                            :base-type    :type/Text
+                                            :special-type :type/Category
+                                            :table-id     1})]
+                     :order-by           [{:field     (qpi/map->Field
+                                                       {:field-id     4
+                                                        :field-name   "text"
+                                                        :base-type    :type/Text
+                                                        :special-type :type/Category
+                                                        :table-id     1})
+                                           :direction :ascending}]}
+      :order-by     [{:field     (qpi/map->FieldLiteral {:field-name "text", :base-type :type/Text, :datetime-unit nil})
+                      :direction :ascending}]})))
diff --git a/test/metabase/query_processor/middleware/fetch_source_query_test.clj b/test/metabase/query_processor/middleware/fetch_source_query_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..c0fc6e9afd8ca39b23a1e0149ef6f1621edf630c
--- /dev/null
+++ b/test/metabase/query_processor/middleware/fetch_source_query_test.clj
@@ -0,0 +1,103 @@
+(ns metabase.query-processor.middleware.fetch-source-query-test
+  (:require [expectations :refer [expect]]
+            [medley.core :as m]
+            [metabase
+             [query-processor :as qp]
+             [util :as u]]
+            [metabase.models
+             [card :refer [Card]]
+             [database :as database]]
+            [metabase.query-processor.middleware.fetch-source-query :as fetch-source-query]
+            [metabase.test.data :as data]
+            [toucan.util.test :as tt]))
+
+(def ^:private ^{:arglists '([query])} fetch-source-query (fetch-source-query/fetch-source-query identity))
+
+;; make sure that the `fetch-source-query` middleware correctly resolves MBQL queries
+(expect
+  {:database (data/id)
+   :type     :query
+   :query    {:aggregation  [:count]
+              :breakout     [[:field-literal :price :type/Integer]]
+              :source-query {:source-table (data/id :venues)}}}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :query
+                                            :query    {:source-table (data/id :venues)}}}]
+    (fetch-source-query {:database database/virtual-id
+                         :type     :query
+                         :query    {:source-table (str "card__" (u/get-id card))
+                                    :aggregation  [:count]
+                                    :breakout     [[:field-literal :price :type/Integer]]}})))
+
+;; make sure that the `fetch-source-query` middleware correctly resolves native queries
+(expect
+  {:database (data/id)
+   :type     :query
+   :query    {:aggregation  [:count]
+              :breakout     [[:field-literal :price :type/Integer]]
+              :source-query {:native        (format "SELECT * FROM %s" (data/format-name "venues"))
+                             :template_tags nil}}}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :native
+                                            :native   {:query (format "SELECT * FROM %s" (data/format-name "venues"))}}}]
+    (fetch-source-query {:database database/virtual-id
+                         :type     :query
+                         :query    {:source-table (str "card__" (u/get-id card))
+                                    :aggregation  [:count]
+                                    :breakout     [[:field-literal :price :type/Integer]]}})))
+
+
+;; test that the `metabase.query-processor/expand` function properly handles nested queries (this function should call `fetch-source-query`)
+(expect
+  {:database     {:name "test-data", :id (data/id), :engine :h2}
+   :type         :query
+   :query        {:source-query {:source-table {:schema "PUBLIC", :name "VENUES", :id (data/id :venues)}
+                                 :join-tables  nil}}
+   :fk-field-ids #{}}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :query
+                                            :query    {:source-table (data/id :venues)}}}]
+    (-> (qp/expand {:database database/virtual-id
+                    :type     :query
+                    :query    {:source-table (str "card__" (u/get-id card))}})
+        (m/dissoc-in [:database :features])
+        (m/dissoc-in [:database :details]))))
+
+;; make sure that nested nested queries work as expected
+(expect
+  {:database (data/id)
+   :type     :query
+   :query    {:limit        25
+              :source-query {:limit        50
+                             :source-query {:source-table (data/id :venues)
+                                            :limit        100}}}}
+  (tt/with-temp* [Card [card-1 {:dataset_query {:database (data/id)
+                                                :type     :query
+                                                :query    {:source-table (data/id :venues), :limit 100}}}]
+                  Card [card-2 {:dataset_query {:database database/virtual-id
+                                                :type     :query
+                                                :query    {:source-table (str "card__" (u/get-id card-1)), :limit 50}}}]]
+    ((fetch-source-query/fetch-source-query identity) {:database database/virtual-id
+                                                       :type     :query
+                                                       :query    {:source-table (str "card__" (u/get-id card-2)), :limit 25}})))
+
+(expect
+  {:database     {:name "test-data", :id (data/id), :engine :h2}
+   :type         :query
+   :query        {:limit        25
+                  :source-query {:limit 50
+                                 :source-query {:source-table {:schema "PUBLIC", :name "VENUES", :id (data/id :venues)}
+                                                :limit        100
+                                                :join-tables  nil}}}
+   :fk-field-ids #{}}
+  (tt/with-temp* [Card [card-1 {:dataset_query {:database (data/id)
+                                                :type     :query
+                                                :query    {:source-table (data/id :venues), :limit 100}}}]
+                  Card [card-2 {:dataset_query {:database database/virtual-id
+                                                :type     :query
+                                                :query    {:source-table (str "card__" (u/get-id card-1)), :limit 50}}}]]
+    (-> (qp/expand {:database database/virtual-id
+                    :type     :query
+                    :query    {:source-table (str "card__" (u/get-id card-2)), :limit 25}})
+        (m/dissoc-in [:database :features])
+        (m/dissoc-in [:database :details]))))
diff --git a/test/metabase/query_processor/middleware/permissions_test.clj b/test/metabase/query_processor/middleware/permissions_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..69eb16e37783b6f79217028df02158a04f320715
--- /dev/null
+++ b/test/metabase/query_processor/middleware/permissions_test.clj
@@ -0,0 +1,118 @@
+(ns metabase.query-processor.middleware.permissions-test
+  "Tests for the middleware that checks whether the current user has permissions to run a given query."
+  (:require [expectations :refer :all]
+            [metabase.api.common :as api]
+            [metabase.models
+             [database :refer [Database]]
+             [permissions :as perms]
+             [permissions-group :as perms-group]
+             [table :refer [Table]]
+             [user :as user]]
+            [metabase.query-processor.middleware.permissions :refer [check-query-permissions]]
+            [metabase.test.data.users :as users]
+            [metabase.util :as u]
+            [toucan.util.test :as tt]))
+
+(def ^:private ^{:arglists '([query]), :style/indent 0} check-perms (check-query-permissions identity))
+
+(defn- do-with-rasta
+  "Call F with rasta as the current user."
+  [f]
+  (binding [api/*current-user-id*              (users/user->id :rasta)
+            api/*current-user-permissions-set* (atom (user/permissions-set (users/user->id :rasta)))]
+    (f)))
+
+(defn- check-perms-for-rasta
+  "Check permissions for QUERY with rasta as the current user."
+  {:style/indent 0}
+  [query]
+  (do-with-rasta (fn [] (check-perms query))))
+
+;;; ------------------------------------------------------------ Native Queries ------------------------------------------------------------
+
+;; Make sure the NATIVE query fails to run if current user doesn't have perms
+(expect
+  Exception
+  (check-perms-for-rasta
+    {:database 1000
+     :type     :native
+     :native   {:query "SELECT * FROM VENUES"}}))
+
+;; ...but it should work if user has perms
+(tt/expect-with-temp [Database [db]]
+  ;; query should be returned by middleware unchanged
+  {:database (u/get-id db)
+   :type     :native
+   :native   {:query "SELECT * FROM VENUES"}}
+  (check-perms-for-rasta
+    {:database (u/get-id db)
+     :type     :native
+     :native   {:query "SELECT * FROM VENUES"}}))
+
+
+;;; ------------------------------------------------------------ MBQL Queries ------------------------------------------------------------
+
+(expect
+  Exception
+  (tt/with-temp* [Database [db]
+                  Table    [table {:db_id (u/get-id db)}]]
+    (perms/revoke-permissions! (perms-group/all-users) (u/get-id db)) ; All users get perms for all new DBs by default
+    (check-perms-for-rasta
+      {:database (u/get-id db)
+       :type     :query
+       :query    {:source-table {:name "Toucans", :id (u/get-id table)}}})))
+
+(tt/expect-with-temp [Database [db]
+                      Table    [table {:db_id (u/get-id db)}]]
+  ;; query should be returned by middleware unchanged
+  {:database (u/get-id db)
+   :type     :query
+   :query    {:source-table {:name "Toucans", :id (u/get-id table)}}}
+  (check-perms-for-rasta
+    {:database (u/get-id db)
+     :type     :query
+     :query    {:source-table {:name "Toucans", :id (u/get-id table)}}}))
+
+
+;;; ------------------------------------------------------------ Nested Native Queries ------------------------------------------------------------
+
+(expect
+  Exception
+  (check-perms-for-rasta
+    {:database 1000
+     :type     :query
+     :query   {:source-query {:native "SELECT * FROM VENUES"}}}))
+
+;; ...but it should work if user has perms
+(tt/expect-with-temp [Database [db]]
+  {:database (u/get-id db)
+   :type     :query
+   :query    {:source-query {:native "SELECT * FROM VENUES"}}}
+  (check-perms-for-rasta
+    {:database (u/get-id db)
+     :type     :query
+     :query   {:source-query {:native "SELECT * FROM VENUES"}}}))
+
+
+;;; ------------------------------------------------------------ Nested MBQL Queries ------------------------------------------------------------
+
+;; For nested queries MBQL make sure perms are checked
+(expect
+  Exception
+  (tt/with-temp* [Database [db]
+                  Table    [table {:db_id (u/get-id db)}]]
+    (perms/revoke-permissions! (perms-group/all-users) (u/get-id db)) ; All users get perms for all new DBs by default
+    (check-perms-for-rasta
+      {:database (u/get-id db)
+       :type     :query
+       :query    {:source-query {:source-table {:name "Toucans", :id (u/get-id table)}}}})))
+
+(tt/expect-with-temp [Database [db]
+                      Table    [table {:db_id (u/get-id db)}]]
+  {:database (u/get-id db)
+   :type     :query
+   :query    {:source-query {:source-table {:name "Toucans", :id (u/get-id table)}}}}
+  (check-perms-for-rasta
+    {:database (u/get-id db)
+     :type     :query
+     :query    {:source-query {:source-table {:name "Toucans", :id (u/get-id table)}}}}))
diff --git a/test/metabase/query_processor/middleware/results_metadata_test.clj b/test/metabase/query_processor/middleware/results_metadata_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..de285c7de8a2bc8b29c3e7f360f0314f9c182fe9
--- /dev/null
+++ b/test/metabase/query_processor/middleware/results_metadata_test.clj
@@ -0,0 +1,104 @@
+(ns metabase.query-processor.middleware.results-metadata-test
+  (:require [expectations :refer [expect]]
+            [metabase
+             [query-processor :as qp]
+             [util :as u]]
+            [metabase.models
+             [card :refer [Card]]
+             [database :as database]]
+            [metabase.query-processor.middleware.results-metadata :as results-metadata]
+            [metabase.test
+             [data :as data]
+             [util :as tu]]
+            [metabase.test.data.users :as users]
+            [toucan.db :as db]
+            [toucan.util.test :as tt]))
+
+(defn- native-query [sql]
+  {:database (data/id)
+   :type     :native
+   :native   {:query sql}})
+
+(defn- card-metadata [card]
+  (db/select-one-field :result_metadata Card :id (u/get-id card)))
+
+;; test that Card result metadata is saved after running a Card
+(expect
+  [{:name "ID",          :display_name "ID",          :base_type "type/Integer"}
+   {:name "NAME",        :display_name "Name",        :base_type "type/Text"}
+   {:name "PRICE",       :display_name "Price",       :base_type "type/Integer"}
+   {:name "CATEGORY_ID", :display_name "Category ID", :base_type "type/Integer"}
+   {:name "LATITUDE",    :display_name "Latitude",    :base_type "type/Float"}
+   {:name "LONGITUDE",   :display_name "Longitude",   :base_type "type/Float"}]
+  (tt/with-temp Card [card]
+    (qp/process-query (assoc (native-query "SELECT ID, NAME, PRICE, CATEGORY_ID, LATITUDE, LONGITUDE FROM VENUES")
+                        :info {:card-id    (u/get-id card)
+                               :query-hash (byte-array 0)}))
+    (card-metadata card)))
+
+;; check that using a Card as your source doesn't overwrite the results metadata...
+(expect
+  {:name "NAME", :display_name "Name", :base_type "type/Text"}
+  (tt/with-temp Card [card {:dataset_query   (native-query "SELECT * FROM VENUES")
+                            :result_metadata {:name "NAME", :display_name "Name", :base_type "type/Text"}}]
+    (qp/process-query {:database database/virtual-id
+                       :type     :query
+                       :query    {:source-table (str "card__" (u/get-id card))}})
+    (card-metadata card)))
+
+;; ...even when running via the API endpoint
+(expect
+  {:name "NAME", :display_name "Name", :base_type "type/Text"}
+  (tt/with-temp Card [card {:dataset_query   (native-query "SELECT * FROM VENUES")
+                            :result_metadata {:name "NAME", :display_name "Name", :base_type "type/Text"}}]
+    ((users/user->client :rasta) :post 200 "dataset" {:database database/virtual-id
+                                                      :type     :query
+                                                      :query    {:source-table (str "card__" (u/get-id card))}})
+    (card-metadata card)))
+
+
+;; tests for valid-checksum?
+(tu/resolve-private-vars metabase.query-processor.middleware.results-metadata metadata-checksum)
+
+(expect
+  (results-metadata/valid-checksum? "ABCDE" (metadata-checksum "ABCDE")))
+
+(expect
+  false
+  (results-metadata/valid-checksum? "ABCD" (metadata-checksum "ABCDE")))
+
+
+;; make sure that queries come back with metadata
+(expect
+  {:checksum java.lang.String
+   :columns [{:base_type :type/Integer, :display_name "ID",          :name "ID"}
+             {:base_type :type/Text,    :display_name "Name",        :name "NAME"}
+             {:base_type :type/Integer, :display_name "Price",       :name "PRICE"}
+             {:base_type :type/Integer, :display_name "Category ID", :name "CATEGORY_ID"}
+             {:base_type :type/Float,   :display_name "Latitude",    :name "LATITUDE"}
+             {:base_type :type/Float,   :display_name "Longitude",   :name "LONGITUDE"}]}
+  (-> (qp/process-query {:database (data/id)
+                         :type     :native
+                         :native   {:query (format "SELECT ID, NAME, PRICE, CATEGORY_ID, LATITUDE, LONGITUDE FROM VENUES")}})
+      (get-in [:data :results_metadata])
+      (update :checksum class)))
+
+;; make sure that a Card where a DateTime column is broken out by year advertises that column as Text, since you can't do datetime breakouts on years
+(expect
+  [{:base_type    "type/Text"
+    :display_name "Date"
+    :name         "DATE"
+    :unit         nil}
+   {:base_type    "type/Integer"
+    :display_name "count"
+    :name         "count"
+    :special_type "type/Number"}]
+  (tt/with-temp Card [card]
+    (qp/process-query {:database (data/id)
+                       :type     :query
+                       :query    {:source-table (data/id :checkins)
+                                  :aggregation  [[:count]]
+                                  :breakout     [[:datetime-field [:field-id (data/id :checkins :date)] :year]]}
+                       :info     {:card-id    (u/get-id card)
+                                  :query-hash (byte-array 0)}})
+    (card-metadata card)))
diff --git a/test/metabase/query_processor_test.clj b/test/metabase/query_processor_test.clj
index 865892b8381974c3907bdc412bce0cf92d38576c..2a50ff5a29aad276555645561ee5b219a0258a7d 100644
--- a/test/metabase/query_processor_test.clj
+++ b/test/metabase/query_processor_test.clj
@@ -8,7 +8,8 @@
              [driver :as driver]
              [util :as u]]
             [metabase.test.data :as data]
-            [metabase.test.data.datasets :as datasets]))
+            [metabase.test.data.datasets :as datasets]
+            [medley.core :as m]))
 
 ;; make sure all the driver test extension namespaces are loaded <3
 ;; if this isn't done some things will get loaded at the wrong time which can end up causing test databases to be created more than once, which fails
@@ -115,6 +116,15 @@
             :name         (data/format-name "name")
             :display_name "Name"})))
 
+(defn- add-min-max
+  "For databases that support binning, min/max values will be
+  populated, otherwise it will remain nil"
+  [min-val max-val original-map]
+  (merge original-map
+         (when (data/binning-supported?)
+           {:min_value min-val
+            :max_value max-val})))
+
 ;; #### users
 (defn users-col
   "Return column information for the `users` column named by keyword COL."
@@ -124,12 +134,11 @@
    {:table_id (data/id :users)
     :id       (data/id :users col)}
    (case col
-     :id         {:special_type :type/PK
-                  :base_type    (data/id-field-type)
-                  :name         (data/format-name "id")
-                  :display_name "ID"
-                  :min_value 1.0
-                  :max_value 15.0}
+     :id         (add-min-max 1.0 15.0
+                              {:special_type :type/PK
+                               :base_type    (data/id-field-type)
+                               :name         (data/format-name "id")
+                               :display_name "ID"})
      :name       {:special_type :type/Name
                   :base_type    (data/expected-base-type->actual :type/Text)
                   :name         (data/format-name "name")
@@ -154,12 +163,11 @@
    {:table_id (data/id :venues)
     :id       (data/id :venues col)}
    (case col
-     :id          {:special_type :type/PK
-                   :base_type    (data/id-field-type)
-                   :name         (data/format-name "id")
-                   :display_name "ID"
-                   :min_value    1.0
-                   :max_value    100.0}
+     :id          (add-min-max 1.0 100.0
+                               {:special_type :type/PK
+                                :base_type    (data/id-field-type)
+                                :name         (data/format-name "id")
+                                :display_name "ID"})
      :category_id {:extra_info   (if (data/fks-supported?)
                                    {:target_table_id (data/id :categories)}
                                    {})
@@ -169,33 +177,24 @@
                                    :type/Category)
                    :base_type    (data/expected-base-type->actual :type/Integer)
                    :name         (data/format-name "category_id")
-                   :display_name "Category ID"
-                   :min_value    nil
-                   :max_value    nil}
-     :price       {:special_type :type/Category
-                   :base_type    (data/expected-base-type->actual :type/Integer)
-                   :name         (data/format-name "price")
-                   :display_name "Price"
-                   :min_value    1.0
-                   :max_value    4.0}
+                   :display_name "Category ID"}
+     :price       (add-min-max 1.0 4.0
+                               {:special_type :type/Category
+                                :base_type    (data/expected-base-type->actual :type/Integer)
+                                :name         (data/format-name "price")
+                                :display_name "Price"})
      :longitude   {:special_type :type/Longitude
                    :base_type    (data/expected-base-type->actual :type/Float)
                    :name         (data/format-name "longitude")
-                   :display_name "Longitude"
-                   :min_value    nil
-                   :max_value    nil}
+                   :display_name "Longitude"}
      :latitude    {:special_type :type/Latitude
                    :base_type    (data/expected-base-type->actual :type/Float)
                    :name         (data/format-name "latitude")
-                   :display_name "Latitude"
-                   :min_value    nil
-                   :max_value    nil}
+                   :display_name "Latitude"}
      :name        {:special_type :type/Name
                    :base_type    (data/expected-base-type->actual :type/Text)
                    :name         (data/format-name "name")
-                   :display_name "Name"
-                   :min_value    nil
-                   :max_value    nil})))
+                   :display_name "Name"})))
 
 (defn venues-cols
   "`cols` information for all the columns in `venues`."
@@ -216,29 +215,27 @@
                 :base_type    (data/id-field-type)
                 :name         (data/format-name "id")
                 :display_name "ID"}
-     :venue_id {:extra_info   (if (data/fks-supported?)
-                                {:target_table_id (data/id :venues)}
-                                {})
-                :target       (target-field (venues-col :id))
-                :special_type (if (data/fks-supported?)
-                                :type/FK
-                                :type/Category)
-                :base_type    (data/expected-base-type->actual :type/Integer)
-                :name         (data/format-name "venue_id")
-                :display_name "Venue ID"
-                :min_value    1.0
-                :max_value    100.0}
-     :user_id  {:extra_info   (if (data/fks-supported?) {:target_table_id (data/id :users)}
-                                  {})
-                :target       (target-field (users-col :id))
-                :special_type (if (data/fks-supported?)
-                                :type/FK
-                                :type/Category)
-                :base_type    (data/expected-base-type->actual :type/Integer)
-                :name         (data/format-name "user_id")
-                :display_name "User ID"
-                :min_value     1.0
-                :max_value     15.0})))
+     :venue_id (add-min-max 1.0 100.0
+                            {:extra_info   (if (data/fks-supported?)
+                                             {:target_table_id (data/id :venues)}
+                                             {})
+                             :target       (target-field (venues-col :id))
+                             :special_type (if (data/fks-supported?)
+                                             :type/FK
+                                             :type/Category)
+                             :base_type    (data/expected-base-type->actual :type/Integer)
+                             :name         (data/format-name "venue_id")
+                             :display_name "Venue ID"})
+     :user_id  (add-min-max 1.0 15.0
+                            {:extra_info   (if (data/fks-supported?) {:target_table_id (data/id :users)}
+                                               {})
+                             :target       (target-field (users-col :id))
+                             :special_type (if (data/fks-supported?)
+                                             :type/FK
+                                             :type/Category)
+                             :base_type    (data/expected-base-type->actual :type/Integer)
+                             :name         (data/format-name "user_id")
+                             :display_name "User ID"}))))
 
 
 ;;; #### aggregate columns
@@ -277,10 +274,14 @@
 (defn breakout-col [column]
   (assoc column :source :breakout))
 
+;; TODO - maybe this needs a new name now that it also removes the results_metadata
 (defn booleanize-native-form
-  "Convert `:native_form` attribute to a boolean to make test results comparisons easier."
+  "Convert `:native_form` attribute to a boolean to make test results comparisons easier.
+   Remove `data.results_metadata` as well since it just takes a lot of space and the checksum can vary based on whether encryption is enabled."
   [m]
-  (update-in m [:data :native_form] boolean))
+  (-> m
+      (update-in [:data :native_form] boolean)
+      (m/dissoc-in [:data :results_metadata])))
 
 (defn format-rows-by
   "Format the values in result ROWS with the fns at the corresponding indecies in FORMAT-FNS.
diff --git a/test/metabase/query_processor_test/aggregation_test.clj b/test/metabase/query_processor_test/aggregation_test.clj
index 12f90194ecc2df927e87ad3bc66dcffca1a70f31..79933406eff8f67ad700803d57d3a3dc17cd913a 100644
--- a/test/metabase/query_processor_test/aggregation_test.clj
+++ b/test/metabase/query_processor_test/aggregation_test.clj
@@ -149,8 +149,8 @@
             (ql/aggregation (ql/avg $price) (ql/count) (ql/sum $price))))))
 
 ;; make sure that multiple aggregations of the same type have the correct metadata (#4003)
-;; (TODO - this isn't tested against Mongo, BigQuery or Presto because those drivers don't currently work correctly with multiple columns with the same name)
-(datasets/expect-with-engines (disj non-timeseries-engines :mongo :bigquery :presto)
+;; (TODO - this isn't tested against Mongo or BigQuery because those drivers don't currently work correctly with multiple columns with the same name)
+(datasets/expect-with-engines (disj non-timeseries-engines :mongo :bigquery)
   [(aggregate-col :count)
    (assoc (aggregate-col :count)
      :display_name    "Count 2"
diff --git a/test/metabase/query_processor_test/breakout_test.clj b/test/metabase/query_processor_test/breakout_test.clj
index f9175a8458a576fbe91f306895cf5ba85ae92592..2d7ac1eb144694aba351f78bc1b8e215c93c1b87 100644
--- a/test/metabase/query_processor_test/breakout_test.clj
+++ b/test/metabase/query_processor_test/breakout_test.clj
@@ -1,14 +1,17 @@
 (ns metabase.query-processor-test.breakout-test
   "Tests for the `:breakout` clause."
-  (:require [metabase.query-processor-test :refer :all]
-            [metabase.query-processor.expand :as ql]
+  (:require [metabase
+             [query-processor-test :refer :all]
+             [util :as u]]
             [metabase.models.field :refer [Field]]
-            [metabase.test.data :as data]
-            [metabase.test.util :as tu]
-            [metabase.util :as u]
+            [metabase.query-processor.expand :as ql]
+            [metabase.test
+             [data :as data]
+             [util :as tu]]
+            [metabase.test.data.datasets :as datasets]
             [toucan.db :as db]))
 
-;; single column
+;;; single column
 (qp-expect-with-all-engines
   {:rows    [[1 31] [2 70] [3 75] [4 77] [5 69] [6 70] [7 76] [8 81] [9 68] [10 78] [11 74] [12 59] [13 76] [14 62] [15 34]]
    :columns [(data/format-name "user_id")
@@ -74,21 +77,21 @@
        booleanize-native-form
        (format-rows-by [int int int])))
 
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   [[10.1 1] [33.1 61] [37.7 29] [39.2 8] [40.8 1]]
   (format-rows-by [(partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
             (ql/aggregation (ql/count))
             (ql/breakout (ql/binning-strategy $latitude :num-bins 20))))))
 
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
  [[10.1 1] [30.5 99]]
   (format-rows-by [(partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
             (ql/aggregation (ql/count))
             (ql/breakout (ql/binning-strategy $latitude :num-bins 3))))))
 
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   [[10.1 -165.4 1] [33.1 -119.7 61] [37.7 -124.2 29] [39.2 -78.5 8] [40.8 -78.5 1]]
   (format-rows-by [(partial u/round-to-decimals 1) (partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
@@ -98,14 +101,14 @@
 
 ;; Currently defaults to 8 bins when the number of bins isn't
 ;; specified
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
  [[10.1 1] [30.1 90] [40.1 9]]
   (format-rows-by [(partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
             (ql/aggregation (ql/count))
             (ql/breakout (ql/binning-strategy $latitude :default))))))
 
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
  [[10.1 1] [30.1 61] [35.1 29] [40.1 9]]
   (tu/with-temporary-setting-values [breakout-bin-width 5.0]
     (format-rows-by [(partial u/round-to-decimals 1) int]
@@ -114,7 +117,7 @@
               (ql/breakout (ql/binning-strategy $latitude :default)))))))
 
 ;; Testing bin-width
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   [[10.1 1] [33.1 25] [34.1 36] [37.1 29] [40.1 9]]
   (format-rows-by [(partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
@@ -122,14 +125,14 @@
             (ql/breakout (ql/binning-strategy $latitude :bin-width 1))))))
 
 ;; Testing bin-width using a float
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
    [[10.1 1] [32.6 61] [37.6 29] [40.1 9]]
   (format-rows-by [(partial u/round-to-decimals 1) int]
     (rows (data/run-query venues
             (ql/aggregation (ql/count))
             (ql/breakout (ql/binning-strategy $latitude :bin-width 2.5))))))
 
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   [[33.0 4] [34.0 57]]
   (tu/with-temporary-setting-values [breakout-bin-width 1.0]
     (format-rows-by [(partial u/round-to-decimals 1) int]
@@ -139,8 +142,16 @@
                                  (ql/> $latitude 20)))
               (ql/breakout (ql/binning-strategy $latitude :default)))))))
 
+(defn- round-binning-decimals [result]
+  (let [round-to-decimal #(u/round-to-decimals 4 %)]
+    (-> result
+        (update :min_value round-to-decimal)
+        (update :max_value round-to-decimal)
+        (update-in [:binning_info :min_value] round-to-decimal)
+        (update-in [:binning_info :max_value] round-to-decimal))))
+
 ;;Validate binning info is returned with the binning-strategy
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   (merge (venues-col :latitude)
          {:min_value 10.0646, :source :breakout,
           :max_value 40.7794, :binning_info {:binning_strategy "num-bins", :bin_width 10.0,
@@ -150,10 +161,11 @@
                       (ql/aggregation (ql/count))
                       (ql/breakout (ql/binning-strategy $latitude :default)))
       (get-in [:data :cols])
-      first))
+      first
+      round-binning-decimals))
 
 ;;Validate binning info is returned with the binning-strategy
-(expect-with-non-timeseries-dbs
+(datasets/expect-with-engines (engines-that-support :binning)
   {:status :failed
    :class Exception
    :error (format "Unable to bin field '%s' with id '%s' without a min/max value"
diff --git a/test/metabase/query_processor_test/date_bucketing_test.clj b/test/metabase/query_processor_test/date_bucketing_test.clj
index 540aafc99ca38ae5c29f333b827e26207585cced..02f65fa184a3aaadab637b41d689f18f0e3885eb 100644
--- a/test/metabase/query_processor_test/date_bucketing_test.clj
+++ b/test/metabase/query_processor_test/date_bucketing_test.clj
@@ -304,6 +304,7 @@
 ;; Don't run the minute tests against Oracle because the Oracle tests are kind of slow and case CI to fail randomly when it takes so long to load the data that the times are
 ;; no longer current (these tests pass locally if your machine isn't as slow as the CircleCI ones)
 (expect-with-non-timeseries-dbs-except #{:bigquery :oracle} 4 (count-of-grouping (checkins:4-per-minute) :minute "current"))
+
 (expect-with-non-timeseries-dbs-except #{:bigquery :oracle} 4 (count-of-grouping (checkins:4-per-minute) :minute -1 "minute"))
 (expect-with-non-timeseries-dbs-except #{:bigquery :oracle} 4 (count-of-grouping (checkins:4-per-minute) :minute  1 "minute"))
 
diff --git a/test/metabase/query_processor_test/expressions_test.clj b/test/metabase/query_processor_test/expressions_test.clj
index 90b94bc3247d2bff94228d1a52683a1b4c0803bc..9b580453b2f2855240d500ba7c0696f309558f14 100644
--- a/test/metabase/query_processor_test/expressions_test.clj
+++ b/test/metabase/query_processor_test/expressions_test.clj
@@ -13,9 +13,10 @@
 ;; Test the expansion of the expressions clause
 (expect
   {:expressions {:my-cool-new-field (qpi/map->Expression {:operator :*
-                                                          :args [{:field-id 10, :fk-field-id nil, :datetime-unit nil,
-                                                                  :binning-strategy nil, :binning-param nil}
-                                                                 20.0]})}}                                            ; 20 should be converted to a FLOAT
+                                                          :args     [{:field-id      10,  :fk-field-id      nil,
+                                                                      :datetime-unit nil, :binning-strategy nil,
+                                                                      :binning-param nil}
+                                                                     20.0]})}}                                            ; 20 should be converted to a FLOAT
   (ql/expressions {} {:my-cool-new-field (ql/* (ql/field-id 10) 20)}))
 
 
diff --git a/test/metabase/query_processor_test/field_visibility_test.clj b/test/metabase/query_processor_test/field_visibility_test.clj
index f9f5b1968e11e9e5b3605bd2905069d7e1a875fb..ef1873baa410a981dcef5a82b51d03e069b2170e 100644
--- a/test/metabase/query_processor_test/field_visibility_test.clj
+++ b/test/metabase/query_processor_test/field_visibility_test.clj
@@ -18,15 +18,11 @@
   [(set (venues-cols))
    #{(venues-col :category_id)
      (venues-col :name)
-     (assoc (venues-col :latitude)
-       :min_value nil
-       :max_value nil)
+     (venues-col :latitude)
      (assoc (venues-col :id)
        :min_value nil
        :max_value nil)
-     (assoc (venues-col :longitude)
-       :min_value nil
-       :max_value nil)
+     (venues-col :longitude)
      (assoc (venues-col :price) :visibility_type :details-only
             :min_value nil
             :max_value nil)}
diff --git a/test/metabase/query_processor_test/nested_queries_test.clj b/test/metabase/query_processor_test/nested_queries_test.clj
new file mode 100644
index 0000000000000000000000000000000000000000..93d8f94e5c77d1f682a8e968fa488a77d97076f7
--- /dev/null
+++ b/test/metabase/query_processor_test/nested_queries_test.clj
@@ -0,0 +1,446 @@
+(ns metabase.query-processor-test.nested-queries-test
+  "Tests for handling queries with nested expressions."
+  (:require [clojure.string :as str]
+            [expectations :refer [expect]]
+            [honeysql.core :as hsql]
+            [metabase
+             [query-processor :as qp]
+             [query-processor-test :refer :all]
+             [util :as u]]
+            [metabase.driver.generic-sql :as generic-sql]
+            [metabase.models
+             [card :refer [Card]]
+             [database :as database]
+             [field :refer [Field]]
+             [table :refer [Table]]]
+            [metabase.test.data :as data]
+            [metabase.test.data.datasets :as datasets]
+            [toucan.db :as db]
+            [toucan.util.test :as tt]))
+
+(defn- rows+cols
+  "Return the `:rows` and relevant parts of `:cols` from the RESULTS.
+   (This is used to keep the output of various tests below focused and manageable.)"
+  {:style/indent 0}
+  [results]
+  {:rows (rows results)
+   :cols (for [col (get-in results [:data :cols])]
+           {:name      (str/lower-case (:name col))
+            :base_type (:base_type col)})})
+
+
+;; make sure we can do a basic query with MBQL source-query
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  {:rows [[1 "Red Medicine"                  4 10.0646 -165.374 3]
+          [2 "Stout Burgers & Beers"        11 34.0996 -118.329 2]
+          [3 "The Apple Pan"                11 34.0406 -118.428 2]
+          [4 "Wurstküche"                   29 33.9997 -118.465 2]
+          [5 "Brite Spot Family Restaurant" 20 34.0778 -118.261 2]]
+   :cols [{:name "id",          :base_type (data/id-field-type)}
+          {:name "name",        :base_type :type/Text}
+          {:name "category_id", :base_type (data/expected-base-type->actual :type/Integer)}
+          {:name "latitude",    :base_type :type/Float}
+          {:name "longitude",   :base_type :type/Float}
+          {:name "price",       :base_type (data/expected-base-type->actual :type/Integer)}]}
+  (format-rows-by [int str int (partial u/round-to-decimals 4) (partial u/round-to-decimals 4) int]
+    (rows+cols
+      (qp/process-query
+        {:database (data/id)
+         :type     :query
+         :query    {:source-query {:source-table (data/id :venues)
+                                   :order-by     [:asc (data/id :venues :id)]
+                                   :limit        10}
+                    :limit        5}}))))
+
+;; TODO - `identifier`, `quoted-identifier` might belong in some sort of shared util namespace
+(defn- identifier
+  "Return a properly formatted *UNQUOTED* identifier for a Table or Field.
+  (This handles DBs like H2 who require uppercase identifiers, or databases like Redshift do clever hacks
+   like prefixing table names with a unique schema for each test run because we're not
+   allowed to create new databases.)"
+  (^String [table-kw]
+   (let [{schema :schema, table-name :name} (db/select-one [Table :name :schema] :id (data/id table-kw))]
+     (name (hsql/qualify schema table-name))))
+  (^String [table-kw field-kw]
+   (db/select-one-field :name Field :id (data/id table-kw field-kw))))
+
+(defn- quote-identifier [identifier]
+  (first (hsql/format (keyword identifier)
+           :quoting (generic-sql/quote-style datasets/*driver*))))
+
+(def ^:private ^{:arglists '([table-kw] [table-kw field-kw])} ^String quoted-identifier
+  "Return a *QUOTED* identifier for a Table or Field. (This behaves just like `identifier`, but quotes the result)."
+  (comp quote-identifier identifier))
+
+;; make sure we can do a basic query with a SQL source-query
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  {:rows [[1 -165.374  4 3 "Red Medicine"                 10.0646]
+          [2 -118.329 11 2 "Stout Burgers & Beers"        34.0996]
+          [3 -118.428 11 2 "The Apple Pan"                34.0406]
+          [4 -118.465 29 2 "Wurstküche"                   33.9997]
+          [5 -118.261 20 2 "Brite Spot Family Restaurant" 34.0778]]
+   :cols [{:name "id",          :base_type :type/Integer}
+          {:name "longitude",   :base_type :type/Float}
+          {:name "category_id", :base_type (data/expected-base-type->actual :type/Integer)}
+          {:name "price",       :base_type (data/expected-base-type->actual :type/Integer)}
+          {:name "name",        :base_type :type/Text}
+          {:name "latitude",    :base_type :type/Float}]}
+  (format-rows-by [int (partial u/round-to-decimals 4) int int str (partial u/round-to-decimals 4)]
+    (rows+cols
+      (qp/process-query
+        {:database (data/id)
+         :type     :query
+         :query    {:source-query {:native (format "SELECT %s, %s, %s, %s, %s, %s FROM %s"
+                                                   (quoted-identifier :venues :id)
+                                                   (quoted-identifier :venues :longitude)
+                                                   (quoted-identifier :venues :category_id)
+                                                   (quoted-identifier :venues :price)
+                                                   (quoted-identifier :venues :name)
+                                                   (quoted-identifier :venues :latitude)
+                                                   (quoted-identifier :venues))}
+                    :order-by     [:asc [:field-literal (keyword (data/format-name :id)) :type/Integer]]
+                    :limit        5}}))))
+
+
+(def ^:private ^:const breakout-results
+  {:rows [[1 22]
+          [2 59]
+          [3 13]
+          [4  6]]
+   :cols [{:name "price", :base_type :type/Integer}
+          {:name "count", :base_type :type/Integer}]})
+
+;; make sure we can do a query with breakout and aggregation using an MBQL source query
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  breakout-results
+  (rows+cols
+    (format-rows-by [int int]
+      (qp/process-query
+        {:database (data/id)
+         :type     :query
+         :query    {:source-query {:source-table (data/id :venues)}
+                    :aggregation  [:count]
+                    :breakout     [[:field-literal (keyword (data/format-name :price)) :type/Integer]]}}))))
+
+;; make sure we can do a query with breakout and aggregation using a SQL source query
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  breakout-results
+  (rows+cols
+    (format-rows-by [int int]
+      (qp/process-query
+        {:database (data/id)
+         :type     :query
+         :query    {:source-query {:native (format "SELECT * FROM %s" (quoted-identifier :venues))}
+                    :aggregation  [:count]
+                    :breakout     [[:field-literal (keyword (data/format-name :price)) :type/Integer]]}}))))
+
+
+(defn- mbql-card-def
+  "Basic MBQL Card definition. Pass kv-pair clauses for the inner query."
+  {:style/indent 0}
+  [& {:as clauses}]
+  {:dataset_query {:database (data/id)
+                   :type     :query
+                   :query    clauses}})
+
+(defn- venues-mbql-card-def
+  "A basic Card definition that returns raw data for the venues test table.
+   Pass additional kv-pair clauses for the inner query as needed."
+  {:style/indent 0}
+  [& additional-clauses]
+  (apply mbql-card-def :source-table (data/id :venues) additional-clauses))
+
+
+(defn- query-with-source-card {:style/indent 1} [card & {:as additional-clauses}]
+  {:database database/virtual-id
+   :type     :query
+   :query    (merge {:source-table (str "card__" (u/get-id card))}
+                    additional-clauses)})
+
+;; Make sure we can run queries using source table `card__id` format. This is the format that is actually used by the frontend;
+;; it gets translated to the normal `source-query` format by middleware. It's provided as a convenience so only minimal changes
+;; need to be made to the frontend.
+(expect
+  breakout-results
+  (tt/with-temp Card [card (venues-mbql-card-def)]
+    (rows+cols
+      (format-rows-by [int int]
+        (qp/process-query
+          (query-with-source-card card
+            :aggregation [:count]
+            :breakout    [[:field-literal (keyword (data/format-name :price)) :type/Integer]]))))))
+
+;; make sure `card__id`-style queries work with native source queries as well
+(expect
+  breakout-results
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :native
+                                            :native   {:query "SELECT * FROM VENUES"}}}]
+    (rows+cols
+      (format-rows-by [int int]
+        (qp/process-query
+          (query-with-source-card card
+            :aggregation [:count]
+            :breakout    [[:field-literal (keyword (data/format-name :price)) :type/Integer]]))))))
+
+
+;; make sure we can filter by a field literal
+(expect
+  {:rows [[1 "Red Medicine" 4 10.0646 -165.374 3]]
+   :cols [{:name "id",          :base_type :type/BigInteger}
+          {:name "name",        :base_type :type/Text}
+          {:name "category_id", :base_type :type/Integer}
+          {:name "latitude",    :base_type :type/Float}
+          {:name "longitude",   :base_type :type/Float}
+          {:name "price",       :base_type :type/Integer}]}
+  (rows+cols
+    (qp/process-query
+      {:database (data/id)
+       :type     :query
+       :query    {:source-query {:source-table (data/id :venues)}
+                  :filter       [:= [:field-literal (data/format-name :id) :type/Integer] 1]}})))
+
+(def ^:private ^:const ^String venues-source-sql
+  (str "(SELECT \"PUBLIC\".\"VENUES\".\"ID\" AS \"ID\", \"PUBLIC\".\"VENUES\".\"NAME\" AS \"NAME\", "
+       "\"PUBLIC\".\"VENUES\".\"CATEGORY_ID\" AS \"CATEGORY_ID\", \"PUBLIC\".\"VENUES\".\"LATITUDE\" AS \"LATITUDE\", "
+       "\"PUBLIC\".\"VENUES\".\"LONGITUDE\" AS \"LONGITUDE\", \"PUBLIC\".\"VENUES\".\"PRICE\" AS \"PRICE\" FROM \"PUBLIC\".\"VENUES\") \"source\""))
+
+;; make sure that dots in field literal identifiers get escaped so you can't reference fields from other tables using them
+(expect
+  {:query  (format "SELECT * FROM %s WHERE \"BIRD.ID\" = 1 LIMIT 10" venues-source-sql)
+   :params nil}
+  (qp/query->native
+    {:database (data/id)
+     :type     :query
+     :query    {:source-query {:source-table (data/id :venues)}
+                :filter       [:= [:field-literal :BIRD.ID :type/Integer] 1]
+                :limit        10}}))
+
+;; make sure that field-literals work as DateTimeFields
+(expect
+  {:query  (str "SELECT * "
+                (format "FROM %s " venues-source-sql)
+                "WHERE parsedatetime(formatdatetime(\"BIRD.ID\", 'YYYYww'), 'YYYYww') = parsedatetime(formatdatetime(?, 'YYYYww'), 'YYYYww') "
+                "LIMIT 10")
+   :params [#inst "2017-01-01T00:00:00.000000000-00:00"]}
+  (qp/query->native
+    {:database (data/id)
+     :type     :query
+     :query    {:source-query {:source-table (data/id :venues)}
+                :filter       [:= [:datetime-field [:field-literal :BIRD.ID :type/DateTime] :week] "2017-01-01"]
+                :limit        10}}))
+
+;; make sure that aggregation references match up to aggregations from the same level they're from
+;; e.g. the ORDER BY in the source-query should refer the 'stddev' aggregation, NOT the 'avg' aggregation
+(expect
+  {:query (str "SELECT avg(\"stddev\") AS \"avg\" FROM ("
+                   "SELECT STDDEV(\"PUBLIC\".\"VENUES\".\"ID\") AS \"stddev\", \"PUBLIC\".\"VENUES\".\"PRICE\" AS \"PRICE\" "
+                   "FROM \"PUBLIC\".\"VENUES\" "
+                   "GROUP BY \"PUBLIC\".\"VENUES\".\"PRICE\" "
+                   "ORDER BY \"stddev\" DESC, \"PUBLIC\".\"VENUES\".\"PRICE\" ASC"
+               ") \"source\"")
+   :params nil}
+  (qp/query->native
+    {:database (data/id)
+     :type     :query
+     :query    {:source-query {:source-table (data/id :venues)
+                               :aggregation  [[:stddev [:field-id (data/id :venues :id)]]]
+                               :breakout     [[:field-id (data/id :venues :price)]]
+                               :order-by     [[[:aggregate-field 0] :descending]]}
+                :aggregation  [[:avg [:field-literal "stddev" :type/Integer]]]}}))
+
+;; make sure that we handle [field-id [field-literal ...]] forms gracefully, despite that not making any sense
+(expect
+  {:query  (format "SELECT \"category_id\" AS \"category_id\" FROM %s GROUP BY \"category_id\" ORDER BY \"category_id\" ASC LIMIT 10" venues-source-sql)
+   :params nil}
+  (qp/query->native
+    {:database (data/id)
+     :type     :query
+     :query    {:source-query {:source-table (data/id :venues)}
+                :breakout     [:field-id [:field-literal "category_id" :type/Integer]]
+                :limit        10}}))
+
+;; Make sure we can filter by string fields
+(expect
+  {:query  (format "SELECT * FROM %s WHERE \"text\" <> ? LIMIT 10" venues-source-sql)
+   :params ["Coo"]}
+  (qp/query->native {:database (data/id)
+                     :type     :query
+                     :query    {:source-query {:source-table (data/id :venues)}
+                                :limit        10
+                                :filter       [:!= [:field-literal "text" :type/Text] "Coo"]}}))
+
+;; Make sure we can filter by number fields
+(expect
+  {:query  (format "SELECT * FROM %s WHERE \"sender_id\" > 3 LIMIT 10" venues-source-sql)
+   :params nil}
+  (qp/query->native {:database (data/id)
+                     :type     :query
+                     :query    {:source-query {:source-table (data/id :venues)}
+                                :limit        10
+                                :filter       [:> [:field-literal "sender_id" :type/Integer] 3]}}))
+
+;; make sure using a native query with default params as a source works
+(expect
+  {:query  "SELECT * FROM (SELECT * FROM PRODUCTS WHERE CATEGORY = 'Widget' LIMIT 10) \"source\" LIMIT 1048576",
+   :params nil}
+  (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                            :type     :native
+                                            :native   {:query         "SELECT * FROM PRODUCTS WHERE CATEGORY = {{category}} LIMIT 10"
+                                                       :template_tags {:category {:name         "category"
+                                                                                  :display_name "Category"
+                                                                                  :type         "text"
+                                                                                  :required     true
+                                                                                  :default      "Widget"}}}}}]
+    (qp/query->native
+      {:database (data/id)
+       :type     :query
+       :query    {:source-table (str "card__" (u/get-id card))}})))
+
+(defn results-metadata {:style/indent 0} [results]
+  (for [col (get-in results [:data :cols])]
+    (u/select-non-nil-keys col [:base_type :display_name :id :name :source :special_type :table_id :unit :datetime-unit])))
+
+;; make sure a query using a source query comes back with the correct columns metadata
+(expect
+  [{:base_type    :type/BigInteger
+    :display_name "ID"
+    :id           [:field-literal "ID" :type/BigInteger]
+    :name         "ID"
+    :source       :fields
+    :special_type :type/PK
+    :table_id     (data/id :venues)}
+   {:base_type    :type/Text
+    :display_name "Name"
+    :id           [:field-literal "NAME" :type/Text]
+    :name         "NAME"
+    :source       :fields
+    :special_type :type/Name
+    :table_id     (data/id :venues)}
+   {:base_type    :type/Integer
+    :display_name "Category ID"
+    :id           [:field-literal "CATEGORY_ID" :type/Integer]
+    :name         "CATEGORY_ID"
+    :source       :fields
+    :special_type :type/FK
+    :table_id     (data/id :venues)}
+   {:base_type    :type/Float
+    :display_name "Latitude"
+    :id           [:field-literal "LATITUDE" :type/Float]
+    :name         "LATITUDE"
+    :source       :fields
+    :special_type :type/Latitude
+    :table_id     (data/id :venues)}
+   {:base_type    :type/Float
+    :display_name "Longitude"
+    :id           [:field-literal "LONGITUDE" :type/Float]
+    :name         "LONGITUDE"
+    :source       :fields
+    :special_type :type/Longitude
+    :table_id     (data/id :venues)}
+   {:base_type    :type/Integer
+    :display_name "Price"
+    :id           [:field-literal "PRICE" :type/Integer]
+    :name         "PRICE"
+    :source       :fields
+    :special_type :type/Category
+    :table_id     (data/id :venues)}]
+  (-> (tt/with-temp Card [card (venues-mbql-card-def)]
+        (qp/process-query (query-with-source-card card)))
+      results-metadata))
+
+;; make sure a breakout/aggregate query using a source query comes back with the correct columns metadata
+(expect
+  [{:base_type    :type/Text
+    :id           [:field-literal "PRICE" :type/Text]
+    :name         "PRICE"
+    :display_name "Price"
+    :source       :breakout}
+   {:base_type    :type/Integer
+    :display_name "count"
+    :name         "count"
+    :source       :aggregation
+    :special_type :type/Number}]
+  (-> (tt/with-temp Card [card (venues-mbql-card-def)]
+        (qp/process-query (query-with-source-card card
+                            :aggregation  [[:count]]
+                            :breakout     [[:field-literal "PRICE" :type/Text]])))
+      results-metadata))
+
+;; make sure nested queries return the right columns metadata for SQL source queries and datetime breakouts
+(expect
+  [{:base_type    :type/DateTime
+    :display_name "Date"
+    :id           [:field-literal "DATE" :type/DateTime]
+    :name         "DATE"
+    :source       :breakout
+    :unit         :day}
+   {:base_type    :type/Integer
+    :display_name "count"
+    :name         "count"
+    :source       :aggregation
+    :special_type :type/Number}]
+  (-> (tt/with-temp Card [card {:dataset_query {:database (data/id)
+                                                :type     :native
+                                                :native   {:query "SELECT * FROM CHECKINS"}}}]
+        (qp/process-query (query-with-source-card card
+                            :aggregation  [[:count]]
+                            :breakout     [[:datetime-field [:field-literal "DATE" :type/DateTime] :day]])))
+      results-metadata))
+
+;; make sure when doing a nested query we give you metadata that would suggest you should be able to break out a *YEAR*
+(expect
+  [{:base_type    :type/Text
+    :display_name "Date"
+    :id           [:field-literal "DATE" :type/Text]
+    :name         "DATE"
+    :source       :breakout
+    :table_id     (data/id :checkins)}
+   {:base_type    :type/Integer
+    :display_name "Count"
+    :id           [:field-literal :count :type/Integer]
+    :name         "count"
+    :source       :fields}]
+  (-> (tt/with-temp Card [card (mbql-card-def
+                                 :source-table (data/id :checkins)
+                                 :aggregation  [[:count]]
+                                 :breakout     [[:datetime-field [:field-id (data/id :checkins :date)] :year]])]
+        (qp/process-query (query-with-source-card card)))
+      results-metadata))
+
+;; make sure using a time interval filter works
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  :completed
+  (tt/with-temp Card [card (mbql-card-def
+                             :source-table (data/id :checkins))]
+    (-> (query-with-source-card card
+          :filter [:time-interval [:field-literal (identifier :checkins :date) :type/DateTime] -30 :day])
+        qp/process-query
+        :status)))
+
+;; make sure that wrapping a field literal in a datetime-field clause works correctly in filters & breakouts
+(datasets/expect-with-engines (engines-that-support :nested-queries)
+  :completed
+  (tt/with-temp Card [card (mbql-card-def
+                             :source-table (data/id :checkins))]
+    (-> (query-with-source-card card
+          :aggregation [[:count]]
+          :filter      [:= [:datetime-field [:field-literal (identifier :checkins :date) :type/Date] :quarter] "2014-01-01T08:00:00.000Z"]
+          :breakout    [[:datetime-field [:field-literal (identifier :checkins :date) :type/Date] :month]])
+        qp/process-query
+        :status)))
+
+;; make sure timeseries queries generated by "drag-to-filter" work correctly
+(expect
+  :completed
+  (tt/with-temp Card [card (mbql-card-def
+                             :source-table (data/id :checkins))]
+    (-> (query-with-source-card card
+          :aggregation [[:count]]
+          :breakout    [[:datetime-field [:field-literal "DATE" :type/Date] :week]]
+          :filter      [:between
+                        [:datetime-field [:field-literal "DATE" :type/Date] :month]
+                        "2014-02-01T00:00:00-08:00"
+                        "2014-05-01T00:00:00-07:00"])
+        qp/process-query
+        :status)))
diff --git a/test/metabase/test/data.clj b/test/metabase/test/data.clj
index 449598a2dcf88664ce7e00dc53e2fa1b576f6e54..1161240b3cfc53283ade0809bf58b6e3e2004e85 100644
--- a/test/metabase/test/data.clj
+++ b/test/metabase/test/data.clj
@@ -167,10 +167,18 @@
   []
   (contains? (driver/features *driver*) :foreign-keys))
 
+(defn binning-supported?
+  "Does the current engine support binning?"
+  []
+  (contains? (driver/features *driver*) :binning))
+
 (defn default-schema [] (i/default-schema *driver*))
 (defn id-field-type  [] (i/id-field-type *driver*))
 
-(defn expected-base-type->actual [base-type]
+(defn expected-base-type->actual
+  "Return actual `base_type` that will be used for the given driver if we asked for BASE-TYPE.
+   Mainly for Oracle because it doesn't have `INTEGER` types and uses decimals instead."
+  [base-type]
   (i/expected-base-type->actual *driver* base-type))
 
 
diff --git a/test/metabase/test/data/bigquery.clj b/test/metabase/test/data/bigquery.clj
index 6c9a831e48f7433675b12b30aacd4acfb90cc038..cd6b667c0793b7c2a1b6cdaa22a6b5e37af6a88d 100644
--- a/test/metabase/test/data/bigquery.clj
+++ b/test/metabase/test/data/bigquery.clj
@@ -21,18 +21,11 @@
 
 (def ^:private ^String normalize-name (comp (u/rpartial s/replace #"-" "_") name))
 
-(defn- get-env-var [env-var]
-  (or (env (keyword (format "mb-bigquery-%s" (name env-var))))
-      (throw (Exception. (format "In order to test BigQuery, you must specify the env var MB_BIGQUERY_%s."
-                                 (s/upper-case (s/replace (name env-var) #"-" "_")))))))
-
 (def ^:private ^:const details
   (datasets/when-testing-engine :bigquery
-    {:project-id    (get-env-var :project-id)
-     :client-id     (get-env-var :client-id)
-     :client-secret (get-env-var :client-secret)
-     :access-token  (get-env-var :access-token)
-     :refresh-token (get-env-var :refresh-token)}))
+    (reduce (fn [acc env-var]
+              (assoc acc env-var (i/db-test-env-var-or-throw :bigquery env-var)))
+            {} [:project-id :client-id :client-secret :access-token :refresh-token])))
 
 (def ^:private ^:const ^String project-id (:project-id details))
 
diff --git a/test/metabase/test/data/datasets.clj b/test/metabase/test/data/datasets.clj
index ddd7c8492f8f7593d6ccab426790ed13934e261f..1804238861afe211acff29b1e771c6378fe061d8 100644
--- a/test/metabase/test/data/datasets.clj
+++ b/test/metabase/test/data/datasets.clj
@@ -43,6 +43,11 @@
                     #{:h2})]
     (when config/is-test?
       (log/info (color/cyan "Running QP tests against these engines: " engines)))
+
+    (when-not (every? all-valid-engines engines)
+      (throw (Exception.
+              (format "Testing on '%s', but the following drivers are not available '%s'"
+                      engines (set (remove all-valid-engines engines))))))
     engines))
 
 
diff --git a/test/metabase/test/data/druid.clj b/test/metabase/test/data/druid.clj
index b3dfd2c4ca8424b1becb49b56f8363bcec64e8ec..bb121d0e518554ad0b14190100e4e96015594bdd 100644
--- a/test/metabase/test/data/druid.clj
+++ b/test/metabase/test/data/druid.clj
@@ -1,7 +1,6 @@
 (ns metabase.test.data.druid
   (:require [cheshire.core :as json]
             [clojure.java.io :as io]
-            [environ.core :refer [env]]
             [metabase.test.data
              [dataset-definitions :as defs]
              [interface :as i]]
@@ -10,10 +9,8 @@
   (:import metabase.driver.druid.DruidDriver))
 
 (defn- database->connection-details [& _]
-  {:host (or (env :mb-druid-host)
-             (throw (Exception. "In order to test Druid, you must specify `MB_DRUID_HOST`.")))
-   :port (Integer/parseInt (or (env :mb-druid-port)
-                               (throw (Exception. "In order to test Druid, you must specify `MB_DRUID_PORT`."))))})
+  {:host (i/db-test-env-var-or-throw :druid :host)
+   :port (Integer/parseInt (i/db-test-env-var-or-throw :druid :port))})
 
 (u/strict-extend DruidDriver
   i/IDriverTestExtensions
diff --git a/test/metabase/test/data/generic_sql.clj b/test/metabase/test/data/generic_sql.clj
index 43f17f1210b07640d1f415a973e1e82e9afe48ae..6123349db11ae55853fdf1ecb6829143f0c3f1c1 100644
--- a/test/metabase/test/data/generic_sql.clj
+++ b/test/metabase/test/data/generic_sql.clj
@@ -224,18 +224,19 @@
    the calls the resulting function with the rows to insert."
   [& wrap-insert-fns]
   (fn [driver {:keys [database-name], :as dbdef} {:keys [table-name], :as tabledef}]
-    (let [spec       (database->spec driver :db dbdef)
-          table-name (apply hx/qualify-and-escape-dots (qualified-name-components driver database-name table-name))
-          insert!    ((apply comp wrap-insert-fns) (partial do-insert! driver spec table-name))
-          rows       (load-data-get-rows driver dbdef tabledef)]
-      (insert! rows))))
+    (jdbc/with-db-connection [conn (database->spec driver :db dbdef)]
+      (.setAutoCommit (jdbc/get-connection conn) false)
+      (let [table-name (apply hx/qualify-and-escape-dots (qualified-name-components driver database-name table-name))
+            insert!    ((apply comp wrap-insert-fns) (partial do-insert! driver conn table-name))
+            rows       (load-data-get-rows driver dbdef tabledef)]
+        (insert! rows)))))
 
 (def load-data-all-at-once!            "Insert all rows at once."                             (make-load-data-fn))
 (def load-data-chunked!                "Insert rows in chunks of 200 at a time."              (make-load-data-fn load-data-chunked))
 (def load-data-one-at-a-time!          "Insert rows one at a time."                           (make-load-data-fn load-data-one-at-a-time))
 (def load-data-chunked-parallel!       "Insert rows in chunks of 200 at a time, in parallel." (make-load-data-fn load-data-add-ids (partial load-data-chunked pmap)))
 (def load-data-one-at-a-time-parallel! "Insert rows one at a time, in parallel."              (make-load-data-fn load-data-add-ids (partial load-data-one-at-a-time pmap)))
-
+;; ^ the parallel versions aren't neccesarily faster than the sequential versions for all drivers so make sure to do some profiling in order to pick the appropriate implementation
 
 (defn default-execute-sql! [driver context dbdef sql]
   (let [sql (some-> sql s/trim)]
@@ -293,27 +294,23 @@
   ;; Exec SQL for creating the DB
   (execute-sql! driver :server dbdef (str (drop-db-if-exists-sql driver dbdef) ";\n"
                                           (create-db-sql driver dbdef)))
-
   ;; Build combined statement for creating tables + FKs
   (let [statements (atom [])]
-
     ;; Add the SQL for creating each Table
     (doseq [tabledef table-definitions]
       (swap! statements conj (drop-table-if-exists-sql driver dbdef tabledef)
              (create-table-sql driver dbdef tabledef)))
-
     ;; Add the SQL for adding FK constraints
     (doseq [{:keys [field-definitions], :as tabledef} table-definitions]
       (doseq [{:keys [fk], :as fielddef} field-definitions]
         (when fk
           (swap! statements conj (add-fk-sql driver dbdef tabledef fielddef)))))
-
     ;; exec the combined statement
     (execute-sql! driver :db dbdef (s/join ";\n" (map hx/unescape-dots @statements))))
-
   ;; Now load the data for each Table
   (doseq [tabledef table-definitions]
-    (load-data! driver dbdef tabledef)))
+    (u/profile (format "load-data for %s %s %s" (name driver) (:database-name dbdef) (:table-name tabledef))
+      (load-data! driver dbdef tabledef))))
 
 (def IDriverTestExtensionsMixin
   "Mixin for `IGenericSQLTestExtensions` types to implement `create-db!` from `IDriverTestExtensions`."
@@ -323,26 +320,28 @@
 
 ;;; ## Various Util Fns
 
+(defn- do-when-testing-engine {:style/indent 1} [engine f]
+  (require 'metabase.test.data.datasets)
+  ((resolve 'metabase.test.data.datasets/do-when-testing-engine) engine f))
+
 (defn execute-when-testing!
   "Execute a prepared SQL-AND-ARGS against Database with spec returned by GET-CONNECTION-SPEC only when running tests against ENGINE.
    Useful for doing engine-specific setup or teardown."
   {:style/indent 2}
   [engine get-connection-spec & sql-and-args]
-  ((resolve 'metabase.test.data.datasets/do-when-testing-engine)
-   engine
-   (fn []
-     (println (u/format-color 'blue "[%s] %s" (name engine) (first sql-and-args)))
-     (jdbc/execute! (get-connection-spec) sql-and-args)
-     (println (u/format-color 'blue "[OK]")))))
+  (do-when-testing-engine engine
+    (fn []
+      (println (u/format-color 'blue "[%s] %s" (name engine) (first sql-and-args)))
+      (jdbc/execute! (get-connection-spec) sql-and-args)
+      (println (u/format-color 'blue "[OK]")))))
 
 (defn query-when-testing!
   "Execute a prepared SQL-AND-ARGS **query** against Database with spec returned by GET-CONNECTION-SPEC only when running tests against ENGINE.
    Useful for doing engine-specific setup or teardown where `execute-when-testing!` won't work because the query returns results."
   {:style/indent 2}
   [engine get-connection-spec & sql-and-args]
-  ((resolve 'metabase.test.data.datasets/do-when-testing-engine)
-   engine
-   (fn []
-     (println (u/format-color 'blue "[%s] %s" (name engine) (first sql-and-args)))
-     (u/prog1 (jdbc/query (get-connection-spec) sql-and-args)
-       (println (u/format-color 'blue "[OK] -> %s" (vec <>)))))))
+  (do-when-testing-engine engine
+    (fn []
+      (println (u/format-color 'blue "[%s] %s" (name engine) (first sql-and-args)))
+      (u/prog1 (jdbc/query (get-connection-spec) sql-and-args)
+        (println (u/format-color 'blue "[OK] -> %s" (vec <>)))))))
diff --git a/test/metabase/test/data/interface.clj b/test/metabase/test/data/interface.clj
index d1fbdb82e10b10c35ebf8ff3ca47d53033cfa461..5a3c89f006f6307de200df501208fb9cb16e19f4 100644
--- a/test/metabase/test/data/interface.clj
+++ b/test/metabase/test/data/interface.clj
@@ -4,6 +4,7 @@
    Objects that implement `IDriverTestExtensions` know how to load a `DatabaseDefinition` into an
    actual physical RDMS database. This functionality allows us to easily test with multiple datasets."
   (:require [clojure.string :as str]
+            [environ.core :refer [env]]
             [metabase
              [db :as db]
              [driver :as driver]
@@ -113,7 +114,7 @@
     "*OPTIONAL*. Return the base type type that is actually used to store `Fields` of BASE-TYPE.
      The default implementation of this method is an identity fn. This is provided so DBs that don't support a given BASE-TYPE used in the test data
      can specifiy what type we should expect in the results instead.
-     For example, Oracle has `INTEGER` data types, so `:type/Integer` test values are instead stored as `NUMBER`, which we map to `:type/Decimal`.")
+     For example, Oracle has no `INTEGER` data types, so `:type/Integer` test values are instead stored as `NUMBER`, which we map to `:type/Decimal`.")
 
   (format-name ^String [this, ^String table-or-field-name]
     "*OPTIONAL* Transform a lowercase string `Table` or `Field` name in a way appropriate for this dataset
@@ -232,3 +233,37 @@
      (for [fielddef (nest-fielddefs dbdef table-name)]
        (update fielddef :field-name flatten-field-name))
      (flatten-rows dbdef table-name)]))
+
+(defn db-test-env-var
+  "Look up test environment var `:ENV-VAR` for the given `:DATABASE-NAME` containing connection related parameters.
+  If no `:default` param is specified and the var isn't found, throw.
+
+     (db-test-env-var :mysql :user) ; Look up `MB_MYSQL_TEST_USER`"
+  ([engine env-var]
+   (db-test-env-var engine env-var nil))
+  ([engine env-var default]
+   (get env
+        (keyword (format "mb-%s-test-%s" (name engine) (name env-var)))
+        default)))
+
+(defn- to-system-env-var-str
+  "Converts the clojure environment variable form (a keyword) to a
+  stringified version that will be specified at the system level
+
+  i.e. :foo-bar -> FOO_BAR"
+  [env-var-kwd]
+  (-> env-var-kwd
+      name
+      (str/replace "-" "_")
+      str/upper-case))
+
+(defn db-test-env-var-or-throw
+  "Same as `db-test-env-var` but will throw an exception if the variable is nil"
+  ([engine env-var]
+   (db-test-env-var-or-throw engine env-var nil))
+  ([engine env-var default]
+   (or (db-test-env-var engine env-var default)
+       (throw (Exception. (format "In order to test %s, you must specify the env var MB_%s_TEST_%s."
+                                  (name engine)
+                                  (str/upper-case (name engine))
+                                  (to-system-env-var-str env-var)))))))
diff --git a/test/metabase/test/data/mysql.clj b/test/metabase/test/data/mysql.clj
index 6db38b8cbc5572dad17276d98d6ba232f573495a..af215eeb7df7c73801de6fc374f86d971e4e686d 100644
--- a/test/metabase/test/data/mysql.clj
+++ b/test/metabase/test/data/mysql.clj
@@ -1,7 +1,6 @@
 (ns metabase.test.data.mysql
   "Code for creating / destroying a MySQL database from a `DatabaseDefinition`."
-  (:require [environ.core :refer [env]]
-            [metabase.test.data
+  (:require [metabase.test.data
              [generic-sql :as generic]
              [interface :as i]]
             [metabase.util :as u])
@@ -19,11 +18,12 @@
    :type/Time       "TIME"})
 
 (defn- database->connection-details [context {:keys [database-name]}]
-  (merge {:host         "localhost"
-          :port         3306
-          :timezone     :America/Los_Angeles
-          :user         (if (env :circleci) "ubuntu"
-                            "root")}
+  (merge {:host         (i/db-test-env-var-or-throw :mysql :host "localhost")
+          :port         (i/db-test-env-var-or-throw :mysql :port 3306)
+          :user         (i/db-test-env-var :mysql :user "root")
+          :timezone     :America/Los_Angeles}
+         (when-let [password (i/db-test-env-var :mysql :password)]
+           {:password password})
          (when (= context :db)
            {:db database-name})))
 
diff --git a/test/metabase/test/data/oracle.clj b/test/metabase/test/data/oracle.clj
index 33d7c0ccbf502c4bdf86c398fea57be7e80a6d86..dc0f7e3c16d1b12c355a4684298bfe81b149e49a 100644
--- a/test/metabase/test/data/oracle.clj
+++ b/test/metabase/test/data/oracle.clj
@@ -2,23 +2,15 @@
   (:require [clojure.java.jdbc :as jdbc]
             [clojure.string :as s]
             [environ.core :refer [env]]
-            [metabase.driver.generic-sql :as sql]
+            [metabase.driver
+             [generic-sql :as sql]
+             oracle]
             [metabase.test.data
              [generic-sql :as generic]
              [interface :as i]]
             [metabase.util :as u])
   (:import metabase.driver.oracle.OracleDriver))
 
-(defn- get-db-env-var
-  " Look up the relevant connection param from corresponding env var or throw an exception if it's not set.
-
-     (get-db-env-var :user) ; Look up `MB_ORACLE_USER`"
-  [env-var & [default]]
-  (or (env (keyword (format "mb-oracle-%s" (name env-var))))
-      default
-      (throw (Exception. (format "In order to test Oracle, you must specify the env var MB_ORACLE_%s."
-                                 (s/upper-case (name env-var)))))))
-
 ;; Similar to SQL Server, Oracle on AWS doesn't let you create different databases;
 ;; We'll create a unique schema (the same as a "User" in Oracle-land) for each test run and use that to keep
 ;; tests from clobbering over one another; we'll also qualify the names of tables to include their DB name
@@ -35,11 +27,11 @@
 
 
 (def ^:private db-connection-details
-  (delay {:host     (get-db-env-var :host)
-          :port     (Integer/parseInt (get-db-env-var :port "1521"))
-          :user     (get-db-env-var :user)
-          :password (get-db-env-var :password)
-          :sid      (get-db-env-var :sid)}))
+  (delay {:host     (i/db-test-env-var-or-throw :oracle :host)
+          :port     (Integer/parseInt (i/db-test-env-var-or-throw :oracle :port "1521"))
+          :user     (i/db-test-env-var-or-throw :oracle :user)
+          :password (i/db-test-env-var-or-throw :oracle :password)
+          :sid      (i/db-test-env-var-or-throw :oracle :sid)}))
 
 
 (def ^:private ^:const field-base-type->sql-type
@@ -79,7 +71,7 @@
           :drop-table-if-exists-sql  (u/drop-first-arg drop-table-if-exists-sql)
           :execute-sql!              generic/sequentially-execute-sql!
           :field-base-type->sql-type (u/drop-first-arg field-base-type->sql-type)
-          :load-data!                generic/load-data-one-at-a-time-parallel!
+          :load-data!                generic/load-data-one-at-a-time! ; Now that connections are reüsed doing this sequentially actually seems to be faster than parallel
           :pk-sql-type               (constantly "INTEGER GENERATED BY DEFAULT AS IDENTITY (START WITH 1 INCREMENT BY 1) NOT NULL") ; LOL
           :qualified-name-components (partial i/single-db-qualified-name-components session-schema)})
 
@@ -108,6 +100,12 @@
 (def ^:private execute-when-testing-oracle!
   (partial generic/execute-when-testing! :oracle dbspec))
 
+(defn- clean-session-schemas! []
+  "Delete any old session users that for some reason or another were never deleted. For REPL usage."
+  (doseq [schema (non-session-schemas)
+          :when  (re-find #"^CAM_" schema)]
+    (execute-when-testing-oracle! (format "DROP USER %s CASCADE" schema))))
+
 (defn- create-session-user!
   {:expectations-options :before-run}
   []
diff --git a/test/metabase/test/data/postgres.clj b/test/metabase/test/data/postgres.clj
index 9c152d0a866e1ca095a20833c31de2d0fa2c3a2a..29a146ca02b1a49745cd80e5b754229bdb8ef82f 100644
--- a/test/metabase/test/data/postgres.clj
+++ b/test/metabase/test/data/postgres.clj
@@ -1,7 +1,6 @@
 (ns metabase.test.data.postgres
   "Code for creating / destroying a Postgres database from a `DatabaseDefinition`."
-  (:require [environ.core :refer [env]]
-            [metabase.test.data
+  (:require [metabase.test.data
              [generic-sql :as generic]
              [interface :as i]]
             [metabase.util :as u])
@@ -21,11 +20,13 @@
    :type/UUID       "UUID"})
 
 (defn- database->connection-details [context {:keys [database-name]}]
-  (merge {:host     "localhost"
-          :port     5432
+  (merge {:host     (i/db-test-env-var-or-throw :postgresql :host "localhost")
+          :port     (i/db-test-env-var-or-throw :postgresql :port 5432)
           :timezone :America/Los_Angeles}
-         (when (env :circleci)
-           {:user "ubuntu"})
+         (when-let [user (i/db-test-env-var :postgresql :user)]
+           {:user user})
+         (when-let [password (i/db-test-env-var :postgresql :password)]
+           {:password password})
          (when (= context :db)
            {:db database-name})))
 
diff --git a/test/metabase/test/data/presto.clj b/test/metabase/test/data/presto.clj
index 65faebfc1d5b0c1196be0d6d7ad39c3c674c05c0..bd2af39d16e8026a579a5a4347e6c64ca065285f 100644
--- a/test/metabase/test/data/presto.clj
+++ b/test/metabase/test/data/presto.clj
@@ -1,6 +1,5 @@
 (ns metabase.test.data.presto
   (:require [clojure.string :as s]
-            [environ.core :refer [env]]
             [honeysql
              [core :as hsql]
              [helpers :as h]]
@@ -13,20 +12,12 @@
 
 (resolve-private-vars metabase.driver.presto execute-presto-query! presto-type->base-type quote-name quote+combine-names)
 
-;;; Helpers
-
-(defn- get-env-var [env-var]
-  (or (env (keyword (format "mb-presto-%s" (name env-var))))
-      (throw (Exception. (format "In order to test Presto, you must specify the env var MB_PRESTO_%s."
-                                 (s/upper-case (s/replace (name env-var) #"-" "_")))))))
-
-
 ;;; IDriverTestExtensions implementation
 
 (defn- database->connection-details [context {:keys [database-name]}]
-  (merge {:host (get-env-var :host)
-          :port (get-env-var :port)
-          :user "metabase"
+  (merge {:host (i/db-test-env-var-or-throw :presto :host)
+          :port (i/db-test-env-var-or-throw :presto :port)
+          :user (i/db-test-env-var-or-throw :presto :user "metabase")
           :ssl  false}
          (when (= context :db)
            {:catalog database-name})))
diff --git a/test/metabase/test/data/redshift.clj b/test/metabase/test/data/redshift.clj
index 8368d3167be68a661914732a20e22f59a57c975a..ac566a427ac417ae4dca52edb1d0538822a76332 100644
--- a/test/metabase/test/data/redshift.clj
+++ b/test/metabase/test/data/redshift.clj
@@ -1,6 +1,5 @@
 (ns metabase.test.data.redshift
   (:require [clojure.string :as s]
-            [environ.core :refer [env]]
             [metabase.driver.generic-sql :as sql]
             [metabase.test.data
              [generic-sql :as generic]
@@ -19,22 +18,12 @@
    :type/Integer    "INTEGER"
    :type/Text       "TEXT"})
 
-(defn- get-db-env-var
-  "Look up the relevant env var for AWS connection details or throw an exception if it's not set.
-
-     (get-db-env-var :user) ; Look up `MB_REDSHIFT_USER`"
-  [env-var & [default]]
-  (or (env (keyword (format "mb-redshift-%s" (name env-var))))
-      default
-      (throw (Exception. (format "In order to test Redshift, you must specify the env var MB_REDSHIFT_%s."
-                                 (s/upper-case (name env-var)))))))
-
 (def ^:private db-connection-details
-  (delay {:host     (get-db-env-var :host)
-          :port     (Integer/parseInt (get-db-env-var :port "5439"))
-          :db       (get-db-env-var :db)
-          :user     (get-db-env-var :user)
-          :password (get-db-env-var :password)}))
+  (delay {:host     (i/db-test-env-var-or-throw :redshift :host)
+          :port     (Integer/parseInt (i/db-test-env-var-or-throw :redshift :port "5439"))
+          :db       (i/db-test-env-var-or-throw :redshift :db)
+          :user     (i/db-test-env-var-or-throw :redshift :user)
+          :password (i/db-test-env-var-or-throw :redshift :password)}))
 
 
 ;; Redshift is tested remotely, which means we need to support multiple tests happening against the same remote host at the same time.
diff --git a/test/metabase/test/data/sqlserver.clj b/test/metabase/test/data/sqlserver.clj
index 6802c1761057556c639204ea01b230cd88eac76a..b43098553fe1a4d31eb2066c722be6043cfc4a97 100644
--- a/test/metabase/test/data/sqlserver.clj
+++ b/test/metabase/test/data/sqlserver.clj
@@ -1,8 +1,6 @@
 (ns metabase.test.data.sqlserver
   "Code for creating / destroying a SQLServer database from a `DatabaseDefinition`."
   (:require [clojure.java.jdbc :as jdbc]
-            [clojure.string :as s]
-            [environ.core :refer [env]]
             [metabase.driver.generic-sql :as sql]
             [metabase.test.data
              [datasets :as datasets]
@@ -35,22 +33,11 @@
 (defn- +suffix [db-name]
   (str db-name \_ @db-name-suffix-number))
 
-(defn- get-db-env-var
-  "Since we run our tests on non-Windows machines, we need to connect to a remote server for running tests.
-   Look up the relevant env var or throw an exception if it's not set.
-
-     (get-db-env-var :user) ; Look up `MB_SQL_SERVER_USER`"
-  [env-var & [default]]
-  (or (env (keyword (format "mb-sql-server-%s" (name env-var))))
-      default
-      (throw (Exception. (format "In order to test SQL Server, you must specify the env var MB_SQL_SERVER_%s."
-                                 (s/upper-case (name env-var)))))))
-
 (defn- database->connection-details [context {:keys [database-name]}]
-  {:host         (get-db-env-var :host)
-   :port         (Integer/parseInt (get-db-env-var :port "1433"))
-   :user         (get-db-env-var :user)
-   :password     (get-db-env-var :password)
+  {:host         (i/db-test-env-var-or-throw :sqlserver :host)
+   :port         (Integer/parseInt (i/db-test-env-var-or-throw :sqlserver :port "1433"))
+   :user         (i/db-test-env-var-or-throw :sqlserver :user)
+   :password     (i/db-test-env-var-or-throw :sqlserver :password)
    :db           (when (= context :db)
                    (+suffix database-name))})
 
diff --git a/test/metabase/test/data/users.clj b/test/metabase/test/data/users.clj
index a1415e76d1298cdd4ff7b1e0eafd169f4de90d9d..b3a9f0a659954c2cf113f0f50dc74fa27f753838 100644
--- a/test/metabase/test/data/users.clj
+++ b/test/metabase/test/data/users.clj
@@ -107,12 +107,14 @@
      (:id (fetch-user username)))))
 
 (defn user->credentials
-  "Return a map with `:email` and `:password` for User with USERNAME.
+  "Return a map with `:username` and `:password` for User with USERNAME.
 
-    (user->credentials :rasta) -> {:email \"rasta@metabase.com\", :password \"blueberries\"}"
+    (user->credentials :rasta) -> {:username \"rasta@metabase.com\", :password \"blueberries\"}"
   [username]
   {:pre [(contains? usernames username)]}
-  (select-keys (user->info username) [:email :password]))
+  (let [{:keys [email password]} (user->info username)]
+    {:username email
+     :password password}))
 
 (def ^{:arglists '([id])} id->user
   "Reverse of `user->id`.
diff --git a/test/metabase/test/data/vertica.clj b/test/metabase/test/data/vertica.clj
index 9edd805a27d37a3ce9f3b64558c0d98060e415b3..08057f53be6a74e3844ac0a3288883b6932b1792 100644
--- a/test/metabase/test/data/vertica.clj
+++ b/test/metabase/test/data/vertica.clj
@@ -1,7 +1,6 @@
 (ns metabase.test.data.vertica
   "Code for creating / destroying a Vertica database from a `DatabaseDefinition`."
-  (:require [environ.core :refer [env]]
-            [metabase.driver.generic-sql :as sql]
+  (:require [metabase.driver.generic-sql :as sql]
             [metabase.test.data
              [generic-sql :as generic]
              [interface :as i]]
@@ -22,16 +21,16 @@
 
 
 (defn- db-name []
-  (or (env :mb-vertica-db)
-      "docker"))
+  (i/db-test-env-var-or-throw :vertica :db "docker"))
 
 (def ^:private db-connection-details
-  (delay {:host     (or (env :mb-vertica-host) "localhost")
+  (delay {:host     (i/db-test-env-var-or-throw :vertica :host "localhost")
+          :port     (Integer/parseInt (i/db-test-env-var-or-throw :vertica :port "5433"))
+          :user     (i/db-test-env-var :vertica :user "dbadmin")
+          :password (i/db-test-env-var :vertica :password)
           :db       (db-name)
-          :port     5433
           :timezone :America/Los_Angeles ; why?
-          :user     (or (env :mb-vertica-user) "dbadmin")
-          :password (env :mb-vertica-password)}))
+          }))
 
 (defn- qualified-name-components
   ([_]                             [(db-name)])
diff --git a/test/metabase/test/integrations/ldap.clj b/test/metabase/test/integrations/ldap.clj
new file mode 100644
index 0000000000000000000000000000000000000000..2cb0e54270fa181ac4f643bc1b124341fe028580
--- /dev/null
+++ b/test/metabase/test/integrations/ldap.clj
@@ -0,0 +1,55 @@
+(ns metabase.test.integrations.ldap
+  (:require [clojure.java.io :as io]
+            [expectations :refer [expect]]
+            [metabase.test.util :as tu])
+  (:import (com.unboundid.ldap.listener InMemoryDirectoryServer InMemoryDirectoryServerConfig InMemoryListenerConfig)
+           com.unboundid.ldap.sdk.schema.Schema
+           com.unboundid.ldif.LDIFReader))
+
+
+(def ^:dynamic *ldap-server*
+  "An in-memory LDAP testing server."
+  nil)
+
+(defn- get-server-config []
+  (doto (InMemoryDirectoryServerConfig. (into-array String ["dc=metabase,dc=com"]))
+          (.addAdditionalBindCredentials "cn=Directory Manager" "password")
+          (.setSchema (Schema/getDefaultStandardSchema))
+          (.setListenerConfigs (into-array InMemoryListenerConfig [(InMemoryListenerConfig/createLDAPConfig "LDAP" 0)]))))
+
+(defn- start-ldap-server! []
+  (with-open [ldif (LDIFReader. (io/file (io/resource "ldap.ldif")))]
+    (doto (InMemoryDirectoryServer. (get-server-config))
+            (.importFromLDIF true ldif)
+            (.startListening))))
+
+(defn get-ldap-port
+  "Get the port for the bound in-memory LDAP testing server."
+  []
+  (.getListenPort *ldap-server*))
+
+(defn do-with-ldap-server
+  "Bind `*ldap-server*` and the relevant settings to an in-memory LDAP testing server and executes `f`."
+  [f]
+  (binding [*ldap-server* (start-ldap-server!)]
+    (try
+      (tu/with-temporary-setting-values [ldap-enabled    true
+                                         ldap-host       "localhost"
+                                         ldap-port       (str (get-ldap-port))
+                                         ldap-bind-dn    "cn=Directory Manager"
+                                         ldap-password   "password"
+                                         ldap-user-base  "dc=metabase,dc=com"
+                                         ldap-group-sync true
+                                         ldap-group-base "dc=metabase,dc=com"]
+        (f))
+      (finally (.shutDown *ldap-server* true)))))
+
+(defmacro with-ldap-server
+  "Bind `*ldap-server*` and the relevant settings to an in-memory LDAP testing server and executes BODY."
+  [& body]
+  `(do-with-ldap-server (fn [] ~@body)))
+
+(defmacro expect-with-ldap-server
+  "Generate a unit test that runs ACTUAL with a bound `*ldap-server*` and relevant settings."
+  [expected actual]
+  `(expect ~expected (with-ldap-server ~actual)))
diff --git a/test/metabase/test/mock/moviedb.clj b/test/metabase/test/mock/moviedb.clj
index 7ff287e37e430ed9f8191f2fbc656a62ef82279c..2847ac2f34bd84954ddfbb74069816661357a96b 100644
--- a/test/metabase/test/mock/moviedb.clj
+++ b/test/metabase/test/mock/moviedb.clj
@@ -58,23 +58,23 @@
 (extend MovieDbDriver
   driver/IDriver
   (merge driver/IDriverDefaultsMixin
-         {:analyze-table       (constantly nil)
-          :describe-database   (fn [_ {:keys [exclude-tables]}]
-                                 (let [tables (for [table (vals moviedb-tables)
-                                                    :when (not (contains? exclude-tables (:name table)))]
-                                                (select-keys table [:schema :name]))]
-                                   {:tables (set tables)}))
-          :describe-table      (fn [_ _ table]
-                                 (-> (get moviedb-tables (:name table))
-                                     (dissoc :fks)))
-          :describe-table-fks  (fn [_ _ table]
-                                 (-> (get moviedb-tables (:name table))
-                                     :fks
-                                     set))
-          :features            (constantly #{:foreign-keys})
-          :details-fields      (constantly [])
-          :table-rows-seq      (constantly [{:keypath "movies.filming.description", :value "If the movie is currently being filmed."}
-                                            {:keypath "movies.description", :value "A cinematic adventure."}])}))
+         {:analyze-table      (constantly nil)
+          :describe-database  (fn [_ {:keys [exclude-tables]}]
+                                (let [tables (for [table (vals moviedb-tables)
+                                                   :when (not (contains? exclude-tables (:name table)))]
+                                               (select-keys table [:schema :name]))]
+                                  {:tables (set tables)}))
+          :describe-table     (fn [_ _ table]
+                                (-> (get moviedb-tables (:name table))
+                                    (dissoc :fks)))
+          :describe-table-fks (fn [_ _ table]
+                                (-> (get moviedb-tables (:name table))
+                                    :fks
+                                    set))
+          :features           (constantly #{:foreign-keys})
+          :details-fields     (constantly [])
+          :table-rows-seq     (constantly [{:keypath "movies.filming.description", :value "If the movie is currently being filmed."}
+                                           {:keypath "movies.description", :value "A cinematic adventure."}])}))
 
 (driver/register-driver! :moviedb (MovieDbDriver.))
 
diff --git a/test/metabase/test/util.clj b/test/metabase/test/util.clj
index b738887fc920eae8a64b9a726db4be18abdd1372..24edbaeeebf1acd76b5f3f056b4652858356431f 100644
--- a/test/metabase/test/util.clj
+++ b/test/metabase/test/util.clj
@@ -217,7 +217,7 @@
   (or (symbol? x)
       (instance? clojure.lang.Namespace x)))
 
-(defn resolve-private-vars* [source-namespace target-namespace symbols]
+(defn ^:deprecated resolve-private-vars* [source-namespace target-namespace symbols]
   {:pre [(namespace-or-symbol? source-namespace)
          (namespace-or-symbol? target-namespace)
          (every? symbol? symbols)]}
@@ -227,11 +227,15 @@
                          (throw (Exception. (str source-namespace "/" symb " doesn't exist!"))))]]
     (intern target-namespace symb varr)))
 
-(defmacro resolve-private-vars
+(defmacro ^:deprecated resolve-private-vars
   "Have your cake and eat it too. This Macro adds private functions from another namespace to the current namespace so we can test them.
 
     (resolve-private-vars metabase.driver.generic-sql.sync
-      field-avg-length field-percent-urls)"
+      field-avg-length field-percent-urls)
+
+   DEPRECATED: Just refer to vars directly using `#'` syntax instead of using this macro.
+
+     (#'some-ns/field-avg-length ...)"
   [namespc & symbols]
   `(resolve-private-vars* (quote ~namespc) *ns* (quote ~symbols)))
 
diff --git a/test_resources/ldap.ldif b/test_resources/ldap.ldif
new file mode 100644
index 0000000000000000000000000000000000000000..9d86c6f417da87d19368da755fdc2b3f303445a0
--- /dev/null
+++ b/test_resources/ldap.ldif
@@ -0,0 +1,76 @@
+dn: dc=metabase,dc=com
+objectClass: top
+objectClass: organization
+objectClass: dcObject
+dc: metabase
+o: Metabase Corporation
+
+dn: ou=People,dc=metabase,dc=com
+objectClass: top
+objectClass: organizationalUnit
+ou: People
+
+dn: cn=John Smith,ou=People,dc=metabase,dc=com
+objectClass: top
+objectClass: person
+objectClass: organizationalPerson
+objectClass: inetOrgPerson
+cn: John Smith
+sn: Smith
+givenName: John
+uid: jsmith1
+mail: John.Smith@metabase.com
+userPassword: strongpassword
+
+dn: cn=Sally Brown,ou=People,dc=metabase,dc=com
+objectClass: top
+objectClass: person
+objectClass: organizationalPerson
+objectClass: inetOrgPerson
+cn: Sally Brown
+sn: Brown
+givenName: Sally
+uid: sbrown20
+mail: sally.brown@metabase.com
+userPassword: 1234
+
+dn: ou=Birds,dc=metabase,dc=com
+objectClass: top
+objectClass: organizationalUnit
+ou: Birds
+
+dn: cn=Rasta Toucan,ou=Birds,dc=metabase,dc=com
+objectClass: top
+objectClass: person
+objectClass: organizationalPerson
+objectClass: inetOrgPerson
+cn: Rasta Toucan
+givenName: Rasta
+sn: Toucan
+uid: rasta
+mail: rasta@metabase.com
+userPassword: blueberries
+
+dn: cn=Lucky Pigeon,ou=Birds,dc=metabase,dc=com
+objectClass: top
+objectClass: person
+objectClass: organizationalPerson
+objectClass: inetOrgPerson
+cn: Lucky Pigeon
+givenName: Lucky
+sn: Pigeon
+uid: lucky
+mail: lucky@metabase.com
+userPassword: notalmonds
+
+dn: ou=Groups,dc=metabase,dc=com
+objectClass: top
+objectClass: organizationalUnit
+ou: Groups
+
+dn: cn=Accounting,ou=Groups,dc=metabase,dc=com
+objectClass: top
+objectClass: groupOfNames
+cn: Accounting
+member: cn=John Smith,ou=People,dc=metabase,dc=com
+member: cn=Rasta Toucan,ou=Birds,dc=metabase,dc=com
diff --git a/webpack.config.js b/webpack.config.js
index c08d75a809864330fc0b7e343b1c358ab20fdf20..d050be9676c3a26c5b592d6c0a1ec7cf969d0db7 100644
--- a/webpack.config.js
+++ b/webpack.config.js
@@ -253,11 +253,15 @@ if (NODE_ENV !== "production") {
     config.output.devtoolModuleFilenameTemplate = '[absolute-resource-path]';
     config.output.pathinfo = true;
 } else {
-    // this is required to ensure we don't minify Chevrotain token identifiers
-    // https://github.com/SAP/chevrotain/tree/master/examples/parser/minification
     config.plugins.push(new webpack.optimize.UglifyJsPlugin({
-        warnings: false,
+        // suppress uglify warnings in production
+        // output from these warnings was causing Heroku builds to fail (#5410)
+        compress: {
+            warnings: false,
+        },
         mangle: {
+            // this is required to ensure we don't minify Chevrotain token identifiers
+            // https://github.com/SAP/chevrotain/tree/master/examples/parser/minification
             except: allTokens.map(function(currTok) {
                 return chevrotain.tokenName(currTok);
             })