composer, putting dependencies on the score

Post on 10-May-2015

10.655 Views

Category:

Technology

3 Downloads

Preview:

Click to see full reader

DESCRIPTION

As PHP projects grow and mature so does their list of dependencies and third party code. Managing all these external packages during development and especially deploy is not an easy task and can be very error prone. Enter Composer, a tool that allows you to keep a consistent list of dependencies and versions across your whole team and all your environments, managing and making discovery of new libraries a breeze. Let's see how Composer can solve all your problems with a simple command line interface and a json configuration file.

TRANSCRIPT

C!"p#$rPutting your dependencies on the score

Rafael Dohms / @rdohms

phot

o cr

edit:

Eli W

hite

Evangelist, Speaker and Contributor.

Developer at WEBclusive.

Enabler at AmsterdamPHP.

Rafael Dohms@rdohms

%$ E&$v'(!r P)(*+a general introduction to Composer

Ev$r,-', *!"p#)./the basic stuff you need to know

Up/r'-)./ (! ' M'$0(r!advanced features for more complex scenarios

1.-)./ ,!2r (2.$discovering and sharing libraries

%$ E&$v'(!r P)(*+

per project system wide

central repository

open acceptance

spread out channels

strict standards

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:PEAR

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:PEAR

SVN Externals

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:PEAR

SVN ExternalsGit Submodules

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:PEAR

SVN ExternalsGit Submodules

vendor management script

Pr!b&$":

I need my team and my deployments to use consistent versions of the dependencies of my project

S!&2()!.:PEAR

SVN ExternalsGit Submodules

vendor management script

C!"p#$r!

A per-project dependency manager that allows you to declare a consistent list of

dependencies and versions for your application, as well as a

consistent way of sharing your libraries and making them

discoverable using packagist.org

Ev$r,-', C!"p#)./

I.0('&&)./ C!"p#$r

$ curl -s http://getcomposer.org/installer | php

$ curl -s http://getcomposer.org/installer | php -- --install-dir=bin

Local (embed)

Global

I.0('&&)./ C!"p#$r

$ curl -s http://getcomposer.org/installer | php

$ curl -s http://getcomposer.org/installer | php -- --install-dir=bin

Local (embed)

Global

$ ln -s /usr/bin/composer.phar /usr/bin/composer3p:

I.0('&&)./ C!"p#$r

I.0('&&)./ C!"p#$r

$ composer.phar --version

Composer version 6573fd3

I.0('&&)./ C!"p#$r

$ composer.phar --version

Composer version 6573fd3

php

K$$p )( 2p-'($-!

K$$p )( 2p-'($-!

$ composer.phar self-update

Updating to version 65e95ed. Downloading: 100%

C!"p#$r 101

C!"p#$r 101

$ cd ~/dev/myproject

C!"p#$r 101

$ cd ~/dev/myproject

$ vim composer.json

{ "require": { "silex/silex": "1.0.*" }, "minimum-stability": "dev"}

C!"p#$r 101

$ cd ~/dev/myproject

$ vim composer.json

{ "require": { "silex/silex": "1.0.*" }, "minimum-stability": "dev"}

note: project root

C!"p#$r 101

$ cd ~/dev/myproject

$ vim composer.json

{ "require": { "silex/silex": "1.0.*" }, "minimum-stability": "dev"}

note: project root

“require”: required packages and versions

adv. ex.: >=1.0.0,<1.2-dev

C!"p#$r 101

$ cd ~/dev/myproject

$ vim composer.json

{ "require": { "silex/silex": "1.0.*" }, "minimum-stability": "dev"}

note: project root

“require”: required packages and versions

adv. ex.: >=1.0.0,<1.2-dev

“minimum-stability”: if you only want stable packages

default: stable

C!"p#$r 101

$ cd ~/dev/myproject

$ vim composer.json

$ composer.phar install

{ "require": { "silex/silex": "1.0.*" }, "minimum-stability": "dev"}

note: project root

“require”: required packages and versions

adv. ex.: >=1.0.0,<1.2-dev

“minimum-stability”: if you only want stable packages

default: stable

Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

- Installing symfony/routing (dev-master) Cloning 6bca82c3ea0d42d750de4f49b22020dfd047dc0f

[...]

- Installing silex/silex (dev-master) Cloning 18e248a277adb061602d2bcabe96011db1c76ec0

symfony/routing suggests installing symfony/config (dev-master)symfony/routing suggests installing symfony/yaml (dev-master)symfony/routing suggests installing doctrine/common (>=2.2,<2.3)[...]silex/silex suggests installing symfony/browser-kit (2.1.*)silex/silex suggests installing symfony/css-selector (2.1.*)silex/silex suggests installing symfony/dom-crawler (2.1.*)Writing lock fileGenerating autoload files

Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

- Installing symfony/routing (dev-master) Cloning 6bca82c3ea0d42d750de4f49b22020dfd047dc0f

[...]

- Installing silex/silex (dev-master) Cloning 18e248a277adb061602d2bcabe96011db1c76ec0

symfony/routing suggests installing symfony/config (dev-master)symfony/routing suggests installing symfony/yaml (dev-master)symfony/routing suggests installing doctrine/common (>=2.2,<2.3)[...]silex/silex suggests installing symfony/browser-kit (2.1.*)silex/silex suggests installing symfony/css-selector (2.1.*)silex/silex suggests installing symfony/dom-crawler (2.1.*)Writing lock fileGenerating autoload files

your dependency’s dependencies

Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

- Installing symfony/routing (dev-master) Cloning 6bca82c3ea0d42d750de4f49b22020dfd047dc0f

[...]

- Installing silex/silex (dev-master) Cloning 18e248a277adb061602d2bcabe96011db1c76ec0

symfony/routing suggests installing symfony/config (dev-master)symfony/routing suggests installing symfony/yaml (dev-master)symfony/routing suggests installing doctrine/common (>=2.2,<2.3)[...]silex/silex suggests installing symfony/browser-kit (2.1.*)silex/silex suggests installing symfony/css-selector (2.1.*)silex/silex suggests installing symfony/dom-crawler (2.1.*)Writing lock fileGenerating autoload files

your dependency

your dependency’s dependencies

Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

- Installing symfony/routing (dev-master) Cloning 6bca82c3ea0d42d750de4f49b22020dfd047dc0f

[...]

- Installing silex/silex (dev-master) Cloning 18e248a277adb061602d2bcabe96011db1c76ec0

symfony/routing suggests installing symfony/config (dev-master)symfony/routing suggests installing symfony/yaml (dev-master)symfony/routing suggests installing doctrine/common (>=2.2,<2.3)[...]silex/silex suggests installing symfony/browser-kit (2.1.*)silex/silex suggests installing symfony/css-selector (2.1.*)silex/silex suggests installing symfony/dom-crawler (2.1.*)Writing lock fileGenerating autoload files

your dependency

your dependency’s dependencies

suggestions of other packages, for further features

P)*42p (+$ ($"p!!Let Composer bootstrap you development

B!!(0(r'pp)./ Pr!5$*(0

$ composer.phar create-project fabpot/silex-skeleton ~/dev/myproject

B!!(0(r'pp)./ Pr!5$*(0

$ composer.phar create-project fabpot/silex-skeleton ~/dev/myproject

Installing fabpot/silex-skeleton (dev-master cc19d406cf3cac253715db92d400992d4f3e1b52) - Installing fabpot/silex-skeleton (dev-master) Cloning master

Created project in one-liner/Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

[...]

symfony/routing suggests installing symfony/yaml (dev-master)[...]

Writing lock fileGenerating autoload files

B!!(0(r'pp)./ Pr!5$*(0

$ composer.phar create-project fabpot/silex-skeleton ~/dev/myproject

Installing fabpot/silex-skeleton (dev-master cc19d406cf3cac253715db92d400992d4f3e1b52) - Installing fabpot/silex-skeleton (dev-master) Cloning master

Created project in one-liner/Installing dependencies - Installing pimple/pimple (dev-master) Cloning d2cfa2f02f50abef65c238747c753a5f6786f6be

[...]

symfony/routing suggests installing symfony/yaml (dev-master)[...]

Writing lock fileGenerating autoload files

myproject/composer.jsoncomposer.lockconfig/console/src/templates/vendor/web/

I w'.( (! *!.(r)b2($ ' pr!5$*(Composer can set that up for you.

$ composer.phar create-project dms/dms --dev ~/dev/oss/dms

B!!(0(r'pp)./ C!.(r)b2()!.0

$ composer.phar create-project dms/dms --dev ~/dev/oss/dms

B!!(0(r'pp)./ C!.(r)b2()!.0

gimme dev packages

$ composer.phar create-project dms/dms --dev ~/dev/oss/dms

B!!(0(r'pp)./ C!.(r)b2()!.0

gimme dev packages

    "require-dev": {        "symfony/symfony": ">=2.1-dev",        "doctrine/orm": "dev-master"    },

$ composer.phar create-project dms/dms --dev ~/dev/oss/dms

B!!(0(r'pp)./ C!.(r)b2()!.0

gimme dev packages

    "require-dev": {        "symfony/symfony": ">=2.1-dev",        "doctrine/orm": "dev-master"    },

“require-dev”: only needed if you are going to contribute

H!w -! I 6.-/&!'- (+$ 6&$0?PSR-0 and the modern autoloader

Composer generates an autoload file for all your

dependencies

Composer generates an autoload file for all your

dependencies

vendor/autoload.php

"autoload": {    "psr-0": { "MyNamespace": "<root>" },

"classmap": ["src/", "lib/", "Something.php"],

"files": ["src/MyLibrary/functions.php"]

},

"autoload": {    "psr-0": { "MyNamespace": "<root>" },

"classmap": ["src/", "lib/", "Something.php"],

"files": ["src/MyLibrary/functions.php"]

},

“autoload”: describes the autoloading needed for your library

"autoload": {    "psr-0": { "MyNamespace": "<root>" },

"classmap": ["src/", "lib/", "Something.php"],

"files": ["src/MyLibrary/functions.php"]

},

“autoload”: describes the autoloading needed for your library

“psr-0”: PSR-0 Compatible libraries

"autoload": {    "psr-0": { "MyNamespace": "<root>" },

"classmap": ["src/", "lib/", "Something.php"],

"files": ["src/MyLibrary/functions.php"]

},

“autoload”: describes the autoloading needed for your library

“classmap”: Old PEAR packages and other libraries

“psr-0”: PSR-0 Compatible libraries

"autoload": {    "psr-0": { "MyNamespace": "<root>" },

"classmap": ["src/", "lib/", "Something.php"],

"files": ["src/MyLibrary/functions.php"]

},

“autoload”: describes the autoloading needed for your library

“classmap”: Old PEAR packages and other libraries

“files”: for php functions or initializations

“psr-0”: PSR-0 Compatible libraries

I.0('&&)./, 2p-'()./ '.- "!v)./ !.how does Composer guarantee consistency

composer.json

“composer.json”: metadata and list of your dependencies.

composer.json

“composer.json”: metadata and list of your dependencies.

composer.lock

composer.json

“composer.json”: metadata and list of your dependencies.

“composer.lock”: existing dependencies and current

commit hashes.

composer.lock

composer.json

composer.lock

composer.json

update

composer.lock

composer.json

update install

composer.lock

composer.json

update install

reads

composer.lock

composer.json

update install

reads

gets latest

composer.lock

composer.json

update install

reads

gets latest

writescomposer.lock

composer.json

update install

reads

gets latest

writes

reads

composer.lock

composer.json

update install

reads

gets latest

writes

reads

compares

composer.lock

composer.json

update install

reads

gets latest

writes

reads

gets locked version

compares

composer.lock

composer.json

D$v$&!p)./ '. App ). ' ($'"?

Commit you composer.lock file into the repository, and use

composer install.

D$v$&!p)./ '. App ). ' ($'"?

Commit you composer.lock file into the repository, and use

composer install.

will ensure everyone is on the same “page”

I’" -$v$&!p)./ ' &)br'r,, +$&p!here are some fields you should care about

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

“name”: this should be unique, pick a good one!

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

“name”: this should be unique, pick a good one!

“type”: will be used for more advanced “custom”installs

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

“name”: this should be unique, pick a good one!

“type”: will be used for more advanced “custom”installs

“license”: very important!

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

“name”: this should be unique, pick a good one!

“type”: will be used for more advanced “custom”installs

“license”: very important!

“support”: point people the right way.

{ "name": "vendor-namespace/package-name", "type": "symfony-bundle", "description": "A sample package for examples", "keywords": ["php", "package"], "homepage": "http://doh.ms", "license": "MIT", "support": { "email": "support@mylib.com", "issues": "http://issues.lib.com" } "target-dir": "/folder/to/install",}

“name”: this should be unique, pick a good one!

“type”: will be used for more advanced “custom”installs

“license”: very important!

“support”: point people the right way.

“target-dir”: great for installing sub-dir splits repositories

ex: Symfony Bundles: /Acme/Bundle/MyBundle

M, *!-$ )0 PHP 5.4 !.&,!managing system dependencies

{ "require": { "php": ">=5.3.3", "ext-ldap": "*" }}

{ "require": { "php": ">=5.3.3", "ext-ldap": "*" }}

“php”: PHP version.

{ "require": { "php": ">=5.3.3", "ext-ldap": "*" }}

“php”: PHP version.

“ext-*”: Presence of selected extension

Up/r'-)./ (! ' M'$0(r!

B2( I .$$- ' 0p$*)6* v$r0)!.version modifiers to the rescue!

"acme/foo": "1.0.x-dev#3ebbe75"

"acme/foo": "@dev"

"acme/foo": "1.0.*@beta"

“#<ref>”: Get this specific commit

"acme/foo": "1.0.x-dev#3ebbe75"

"acme/foo": "@dev"

"acme/foo": "1.0.*@beta"

“#<ref>”: Get this specific commit

“@<state>”: Get a version outside your default stability

I .$$- (! 7$*2($ ' f$w 0*r)p(0how to automate tasks with Composer

    "scripts": {        "post-install-cmd": [            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::buildBootstrap",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::clearCache",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installAssets",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installRequirementsFile"        ],        "post-update-cmd": [            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::buildBootstrap",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::clearCache",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installAssets",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installRequirementsFile"        ]    },

    "scripts": {        "post-install-cmd": [            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::buildBootstrap",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::clearCache",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installAssets",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installRequirementsFile"        ],        "post-update-cmd": [            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::buildBootstrap",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::clearCache",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installAssets",            "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installRequirementsFile"        ]    },

“scripts”: allows you to run scripts at given moments

C20(!" R$p#)(!r)$0 !r .! C!"p#$r, w+'( .!w?

Injecting Composer into wild packages

“hero/superpackage”: “dev-master”

“hero/superpackage”: “dev-master”

“hero/superpackage”: “dev-master”

“hero/superpackage”: “dev-master”

“hero/superpackage”: “dev-master”

    "repositories": [ { "type": "vcs", "url": "https://github.com/rdohms/hero-superpackage" }       

]

“hero/superpackage”: “dev-master”

    "repositories": [ { "type": "vcs", "url": "https://github.com/rdohms/hero-superpackage" }       

]

“repositories”: point to non-indexed, override existing or on-the-fly packages

{ "repositories": [ { "type": "package", "package": { "name": "smarty/smarty", "version": "3.1.7", "dist": { "url": "http://www.smarty.net/files/Smarty-3.1.7.zip", "type": "zip" }, "source": { "url": "http://smarty-php.googlecode.com/svn/", "type": "svn", "reference": "tags/Smarty_3_1_7/distribution/" } } } ]}

N!.-C!"p#$r P'*4'/$

{ "repositories": [ { "type": "package", "package": { "name": "smarty/smarty", "version": "3.1.7", "dist": { "url": "http://www.smarty.net/files/Smarty-3.1.7.zip", "type": "zip" }, "source": { "url": "http://smarty-php.googlecode.com/svn/", "type": "svn", "reference": "tags/Smarty_3_1_7/distribution/" } } } ]}

“package”: on-the-fly package, injecting a composer.json

N!.-C!"p#$r P'*4'/$

{ "repositories": [ { "type": "package", "package": { "name": "smarty/smarty", "version": "3.1.7", "dist": { "url": "http://www.smarty.net/files/Smarty-3.1.7.zip", "type": "zip" }, "source": { "url": "http://smarty-php.googlecode.com/svn/", "type": "svn", "reference": "tags/Smarty_3_1_7/distribution/" } } } ]}

SVN / Git

“package”: on-the-fly package, injecting a composer.json

N!.-C!"p#$r P'*4'/$

I .$$- (+)0 PEAR p'*4'/$...No Problem!

{ "repositories": [ { "type": "pear", "url": "http://pear2.php.net" } ], "require": { "pear-pear2.php.net/PEAR2_Text_Markdown": "*", "pear-pear2/PEAR2_HTTP_Request": "*" }}

{ "repositories": [ { "type": "pear", "url": "http://pear2.php.net" } ], "require": { "pear-pear2.php.net/PEAR2_Text_Markdown": "*", "pear-pear2/PEAR2_HTTP_Request": "*" }}

“pear”: official PEAR and custom PEAR channels

{ "repositories": [ { "type": "pear", "url": "http://pear2.php.net" } ], "require": { "pear-pear2.php.net/PEAR2_Text_Markdown": "*", "pear-pear2/PEAR2_HTTP_Request": "*" }}

“pear”: official PEAR and custom PEAR channels

Remember the prefix!

{ "repositories": [ { "type": "pear", "url": "http://pear2.php.net" } ], "require": { "pear-pear2.php.net/PEAR2_Text_Markdown": "*", "pear-pear2/PEAR2_HTTP_Request": "*" }}

“pear”: official PEAR and custom PEAR channels

Remember the prefix!

Warning: PEAR causes a overhead of requests! !

replace provide

alias

replace provide

alias

{ "extra": { "branch-alias": { "dev-master": "1.0.x-dev" } },

"require": { "monolog/monolog": "dev-bugfix as 1.0.x-dev" }}

replace provide

alias

{ "extra": { "branch-alias": { "dev-master": "1.0.x-dev" } },

"require": { "monolog/monolog": "dev-bugfix as 1.0.x-dev" }}

“replace”: allows you to replace other packages, and be used them in their place.

replace provide

alias

{ "extra": { "branch-alias": { "dev-master": "1.0.x-dev" } },

"require": { "monolog/monolog": "dev-bugfix as 1.0.x-dev" }}

“replace”: allows you to replace other packages, and be used them in their place.

“provide”: allows you to say a package provides a

expectation.

1.-)./ ,!2r (2.$

I .$$- ' &)br'r, (+'( -!$0..Let me get that for you

http://packagist.org/

versions

package info

usage info

$ composer.phar search filter

dms/dms-filter-bundle : DMS Filter Bundle, makes Annotation based ...lexik/form-filter-bundle : This bundle aim to provide classes to build...rollerworks/recordfilter-bundle : Record search-filtering bundle for Symfonybrikou/zend_filter : Zend Framework Filter Libraryext-filter : The filter PHP extensiondms/dms-filter : DMS Library, includes various bundles and ...shtumi/useful-bundle : Symfony ShtumiUsefulBundle

$ composer.phar search filter

$ composer.phar show dms/dms-filter-bundle

name : dms/dms-filter-bundledescrip. : DMS Filter Bundle, makes Annotation based entity filtering available in Symfonykeywords : symfony, bundle, filter, dmsversions : dev-master, v1.1.1, v1.1, 1.0.2, 1.0.1, 1.0.0type : symfony-bundlelicense : MITsource : [git] https://github.com/rdohms/DMSFilterBundle v1.1.1dist : [zip] https://github.com/rdohms/DMSFilterBundle/zipball/v1.1.1 v1.1.1names : dms/dms-filter-bundle

autoloadpsr-0DMS\Bundle\FilterBundle => .

requiresphp >=5.3.2dms/dms-filter >=1.0.2

$ composer.phar show dms/dms-filter-bundle

B2( ", r$p#)(!r, )0 pr)v'($!Get your own package repository

S'()0!

$ composer.phar create-project composer/satis

R!&&!2( ,!2r !w. S'()0

$ composer.phar create-project composer/satis

$ vi packages.json

{ "name": "My Repository", "homepage": "http://packages.example.org", "repositories": [ { "type": "vcs", "url": "http://github.com/mycompany/privaterepo" }, { "type": "vcs", "url": "http://svn.example.org/private/repo" }, { "type": "vcs", "url": "http://github.com/mycompany/privaterepo2" } ], "require-all": true}

R!&&!2( ,!2r !w. S'()0

$ composer.phar create-project composer/satis

$ vi packages.json

$ php bin/satis build config.json web/

{ "name": "My Repository", "homepage": "http://packages.example.org", "repositories": [ { "type": "vcs", "url": "http://github.com/mycompany/privaterepo" }, { "type": "vcs", "url": "http://svn.example.org/private/repo" }, { "type": "vcs", "url": "http://github.com/mycompany/privaterepo2" } ], "require-all": true}

R!&&!2( ,!2r !w. S'()0

{ "repositories": [ { "type": "composer", "url": "http://packages.yourdomain.net" } ],

“require”: { “myvendor/mypackage”: “dev-master” }}

U0)./ ,!2r !w. S'()0

{ "repositories": [ { "type": "composer", "url": "http://packages.yourdomain.net" } ],

“require”: { “myvendor/mypackage”: “dev-master” }}

U0)./ ,!2r !w. S'()0“composer”: use this just like it was Packagist

W+$r$ 0+!2&- I /$( +$&p?

http://getcomposer.org

#composer on irc.freenode.org

%$ E&$v'(!r P)(*+Dependency Manager, consistent versions, per-project

Ev$r,-', *!"p#)./install, update, lock and autoload

Up/r'-)./ (! ' M'$0(r!post-install, overriding, PEAR integration, developer environment

1.-)./ ,!2r (2.$Satis and Packagist

%'.4 ,!2!

http://slides.doh.ms

http://doh.ms

@rdohms

https://joind.in/7051

top related