hedronium / jables
Write your Laravel Database Schema in JSON.
Requires
- justinrainbow/json-schema: ~1.3
- seld/jsonlint: ^1.3
- symfony/yaml: ~3.1
This package is not auto-updated.
Last update: 2024-11-09 18:20:56 UTC
README
For Laravel 5
Write your Database Schema in JSON, with clean naming conventions and store them nicely in Version Control.
Features
✓ Schema in JSON
✓ Laravel Integration
✓ All Laravel Field Types supported.
✓ Checking JSON Syntax
✓ Checking Foreign key References.
✓ Checking Unique key Constraints.
✓ Automatic Table Deconstruction.
Soon to come:
- Schema Diff (build changes not complete reconstructions)
- Automatic Documentation Builder
- JSON to Migration Transpiler
Installation
Grab it through Composer.
composer require hedronium/jables
You must add Jables to Laravel as a service provider. To do this, open up your config/app.php
and add hedronium\Jables\JablesServiceProvider
to the Service Providers list. Like:
'providers' => [ // .... other providers and stuff ... hedronium\Jables\JablesServiceProvider::class ]
Testing Installation
On the command line run
php artisan
and check to see if the jables
command and the jables
section shows up.
Usage
Commands
php artisan jables
- Checks your JSON files then creates your Database Tables
php artisan jables:check
- Checks your JSON files and reports errors.
php artisan jables:refresh
- Destroys all the tables then recreates them from your (possibly updated) JSON files. (warning: risk of data loss)
php artisan jables:destroy
- Removes all the tables that jables created from Database.
php artisan jables:create-folder
- Creates the folder to store your Schema based on your configuration.
php artisan jables:create-table
- Creates Jables own tracking table in database.
All the commands accept the --database=[connection]
option. You can use it to override which connection Jables uses to do its business.
example
php artisan jables --database=memory
Writing Schemas
The Schema files are usually stored in the database/jables
(you need to create it) folder unless you configure it to be otherwise. The Filename is your table name. So if you were to create a users
table, your file name would be users.json
nested under database/jables
Getting Started
food.json
{ "fields": { "name": { "type": "string" } } }
now run php artisan jables
. This will create a table named food
with a field named name
with the type being varchar
.
The Formal Breakdown
Well you define all your fields in the fields
property on your root object of your json file. The fields
property itself is an object and every property of the fields
object corresponds to a table field.
Each property (which are the field definitions) within the fields
object is once again another object. The only hard requirement for it all is the type
property on them. This tells jables what is the type of the field.
in our 'hello world' example the type of string
corresponds to varchar
just like in Laravel migrations (Jables uses Laravel's Schema Builder).
Types Available
Heres a list
- big-integer
- binary
- boolean
- char
- date
- date-time
- decimal
- double
- enum
- float
- integer
- json
- jsonb
- long-text
- medium-integer
- medium-text
- morphs
- small-integer
- string
- text
- time
- tiny-integer
- timestamp
integer
"awesomeness": { "type": "integer" }
You can write attributes
which is a list. Currently only supports one attribute, the unsigned
attribute
like...
{ "type": "integer", "attributes": [ "unsigned" ] }
You can only set it to auto-increment with the ai
(boolean
) property like...
{ "type": "integer", "ai" : true }
big-integer
Same as the average integer just write the type different like...
{ "type": "big-integer" }
medium-integer
Same as the average integer just write the type different like...
{ "type": "medium-integer" }
small-integer
Same as the average integer just write the type different like...
{ "type": "small-integer" }
tiny-integer
Same as the average integer just write the type different like...
{ "type": "tiny-integer" }
float
The FLOAT
type equivalent.
{ "type": "float" }
double
The DOUBLE
type equivalent. It requires you to set the digits
& precision
properties.
{ "type": "double", "digits": 10, "precision": 5 }
decimal
The DECIMAL
type. Properties same as double
.
{ "type": "decimal", "digits": 10, "precision": 5 }
string
string
is the VARCHAR
type, and it accepts a length
property like...
{ "type": "string", "length": 50 }
but the length
property isn't required.
char
Its exactly like string it just uses the CHAR
type but the length
property is absolutely required.
{ "type": "char", "length": 10 }
text
Text doesn't require any special properties.
{ "type": "text" }
long-text
Same as text
.
{ "type": "long-text" }
medium-text
Same as text
.
{ "type": "medium-text" }
date
No Special Properties.
{ "type": "date" }
time
No Special Properties.
{ "type": "time" }
date-time
No Special Properties.
{ "type": "date-time" }
timestamp
No Special Properties.
{ "type": "timestamp" }
enum
for the ENUM
type. It is required that you set the values
(list
) property.
{ "type": "enum", "values": ["wizard", "muggle"] }
boolean
No special properties.
{ "type": "boolean" }
json
No special properties.
{ "type": "json" }
jsonb
No special properties.
{ "type": "jsonb" }
morphs
No special properties.
{ "type": "morphs" }
binary
No special properties.
{ "type": "binary" }
Timestamps
Just like in Schema Builder, you can create the two fields created_at
and updated_at
in a simple way.
Create a special timestamps
property in yours fields
object and set it to true.
Like:
{ "fields": { "user_id": { "type": "integer", "attributes": [ "unsigned" ] }, "burger_id": { "type": "integer", "attributes": [ "unsigned" ] }, "timestamps": true } }
softDeletes
Same with softDeletes.
Create a special soft-deletes
property in yours fields
object and set it to true.
Like:
{ "fields": { "user_id": { "type": "integer", "attributes": [ "unsigned" ] }, "soft-deletes": true } }
Default Values
All field definitions accept the default
property for when you want to set the default value of a field.
Used like...
{ "type": "string", "default": "cake" }
Nullable Fields
All field definitions accept the nullable
(boolean
) property. If set to true, the field can be left null.
Used like...
{ "type": "string", "nullable": true }
Primary Keys
if you set the ai
to true on a integer
type or similar field. That field automatically becomes the primary key (its a Laravel thing).
Apart from that, you can set the primary
property on any field to true like...
{ "type": "string", "primary": true }
Composite Primary Keys
More that one field makes your primary key? No Problem! Just create a primary
(list
) property on your root object (sibling to your fields
property) like...
{ "fields": { "region": { "type": "string" }, "house": { "type": "string" } }, "primary": ["region", "house"] }
Unique Constraints
All field definitions accept the unique
property. set it to true
to make it an unique field like...
{ "type": "string", "length": 20, "unique": true }
Composite Unique Constraint
You can created unique constraints across many fields. Just create a unique
(list
) property on your root object (sibling to your fields
property) like...
{ "fields": { "region": { "type": "string" }, "house": { "type": "string" } }, "unique": [ ["region", "house"] ] }
Yes, it is a list inside a list, for when you want multiple composite constraints.
Foreign Key Constraints
Got you covered! All fields accept the foreign
property. You can set it to a string containing the name of the table and the name of the field of that table separated by a dot. (eg. users.id
)
"user_id": { "type": "integer", "attributes": [ "unsigned" ], "foreign": "users.id" }
this user_id
field will now reference id
on the users
table.
You could also define them like you define unique constraints like...
{ "fields": { "user_id": { "type": "integer", "attributes": [ "unsigned" ] }, "burger_id": { "type": "integer", "attributes": [ "unsigned" ] } }, "foreign": { "user_id": "users.id", "burger_id": "burgers.id" } }
This will work totally fine.
Documenting Tables
All field and table definitions accept title
and description
properties that can be used to document your database schema.
We're also working on a feature that generates HTML documentation from your JSON files but for now, you gotta create it on your own. (Maybe even send us a pull request. PLEASE. WE'RE DESPERATE.)
example:
{ "title": "Food", "description": "FOOOD! GLORIOUS FOOD!", "fields": { "name": { "type": "string", "title": "Name", "description": "Name of the food." } } }
Configuration
Jables usually works right out of the box with no configuration required, but two option does exist.
First publish the configuration files. with
php artisan vendor:publish
after running that a jables.php
should show up in your config
folder with the following contents...
<?php return [ 'table' => 'jables', 'folder' => 'jables' ];
Options
table
- The name of the special table jables creates for tracking.folder
- The name of the folder within which you store your table schemas. The name is relative to your Laravel installation'sdatabase
folder.
It is recommend that you store your schema in a folder within your application's
database
folder.