Download the PHP package 8fold/php-shoop without Composer

On this page you can find all versions of the php package 8fold/php-shoop. It is possible to download/install these versions without Composer. Possible dependencies are resolved automatically.

FAQ

After the download, you have to make one include require_once('vendor/autoload.php');. After that you have to import the classes with use statements.

Example:
If you use only one package a project is not needed. But if you use more then one package, without a project it is not possible to import the classes with use statements.

In general, it is recommended to use always a project to download your libraries. In an application normally there is more than one library needed.
Some PHP packages are not free to download and because of that hosted in private repositories. In this case some credentials are needed to access such packages. Please use the auth.json textarea to insert credentials, if a package is coming from a private repository. You can look here for more information.

  • Some hosting areas are not accessible by a terminal or SSH. Then it is not possible to use Composer.
  • To use Composer is sometimes complicated. Especially for beginners.
  • Composer needs much resources. Sometimes they are not available on a simple webspace.
  • If you are using private repositories you don't need to share your credentials. You can set up everything on our site and then you provide a simple download link to your team member.
  • Simplify your Composer build process. Use our own command line tool to download the vendor folder as binary. This makes your build process faster and you don't need to expose your credentials for private repositories.
Please rate this library. Is it a good library?

Informations about the package php-shoop

8fold Shoop for PHP

Shoop is a horizontally-consistent interface into PHP, whereas PHP could be described as a vertically-consistent interface into C.

Shoop is built on 8fold Foldable allowing for ubiquitous construction of data objects.

Installation

Usage

Contrived examples...live examples coming soon and available in the tests folder.

Apply a single filter.

Pipe multiple filters.

Nesting pipes and filters. (Variation on part of the PlusAt Filter.)

Fluent using method chaining.

Types and type juggling

Shoop defines abstract and concrete types.

Abstract Shoop types map directly to PHP types.

Type name PHP type(s)
Content PHP boolean, float, integer, and string.
Collection PHP array, stdClass, object without public methods.
List PHP array.
Sequential PHP indexed array, integer, string.
Object PHP object that can only be juggled from, not to.

Concrete Shoop types:

  1. MUST be self-defined as what they are or are like as opposed to what they are not.
  2. MUST juggle reasonably to all other Shoop concrete types except object.
Type name Abstract type(s) Definition
Boolean Content same as PHP
Number '' all real numbers
Integer '' all whole numbers
String '' same as PHP
Dictionary Collection, List a PHP array with string keys (associative array)
Array Collection, List a PHP array with integer indeces (indexed array) in sequential order starting from 0
Tuple Collection a PHP object containing only public properties with non-null values
JSON Content, Collection see String and Tuple
Object Object see Tuple, with at least one defined public method - or implementing Shoop type interfaces

Note: Rather than being implemented as PHP classes, Shoop types are interpretations of PHP types you implement. Shoop types facilitate type juggling, transformations, and the application of filters.

Filters

Filters are PHP classes inheriting from the Shoop abstract filter and implementing the interface methods required by that inheritance.

Filters generally act as a bridge between Shoopland and PHP. They can be viewed as low-level functions used to manipulate PHP types. For a manipulation to become a filter, it MUST meet AT LEAST three of the following:

  1. Used at least three times in one or more production project(s), which MAY include Shoop itself. ex. Reversed::fromBoolean
  2. Fully testing the proposed filter results in testing multiple other filters. ex. IsEmpty::fromTuple
  3. The proposed filter DOES NOT require testing because it uses PHP directly. ex. IsEmpty::fromString
  4. The proposed filter DOES NOT require testing because it uses approved filters that do not require testing. ex. AsBoolean::*

Note: If the fourth item is used as a reason for the proposed filter to be approved, it SHOULD be accompanied by a specific reason. For example, the AsBoolean filter facilitates juggling to that concrete type and returns the result of IsNotEmpty, which returns the result of IsEmpty that is Reversed. Neither AsBoolean nor IsNotEmpty require testing; however, it is a DRY way of implementing the PHP bool casting operation for most PHP types.

By complying with the previous, the following requirement should be met automatically:

  1. The proposed filter MUST be fully tested, directly or inderectly.

Note: With Shoop, objects are fully specifiable PHP types, as you can implement their representation as any and all Shoop-supported PHP types.

Deviations from PHP

In order to make Shoop easy to adopt we try not to deviate too much from what you're accustomed to. The following details the known deviations from PHP standard behavior:

Shoop PHP
You can transform (cast) any Shoop type to any other Shoop type, except object. Some castings result in error; array to string, for example.
Interfaces exist to define the PHP type representation of custom types or instances implementing the interface. PHP is limited in this regard: see the Stringable and JsonSerializable interfaces as well as some magic methods.
Boolean as string returns the english-equivalent. ex. "true" PHP converts to an integer, then stringify that result. ex. "1"
Boolean as dictionary returns a two-key dictionary holding both true and false values. PHP uses the boolean value as the first index.
Boolean as array uses the array values of the dictionary where zero holds the value of false. PHP does not differentiate between dictionaries and arrays.
JSON can be converted to a dictionary but not created that way. The transformation is non-recursive; so, inner objects remain objects. The string representation MUST start and end with opening and closing curly braces, respectively. The PHP json_decode() function can return a PHP associative array where inner objects are converted to associative arrays. The string representation can start and end using opening and closing square brackets.

Performance

Each test we write covers performance for speed and memory usage.

When it comes to speed, our goal is that every filter can be applied and returned in less than one millisecond (represented as 1.0 in our tests).

Our starting point for speed checks is one microsecond (0.001 in our tests). If a test goes over one microsecond, but less than 5 microseconds, we bump the speed check by one microsecond (ex. 0.001001 becomes 0.002) while leaving the previous, shorter run(s) as comment notes. If a test goes over five microseconds (0.005) we switch to 10 microsecond intervals (ex. 0.005 becomes 0.01). In all other cases, we round up to the nearest 10 microsecond mark (ex. 0.010001 becomes 0.02).

When it comes to memory, our goal is that every filter can be applied and returned using less than 1 kilobyte of memory (1 in our tests). This will be based somewhat on what is passed to the filter. Our tests are measured with a bare miminum value passed in to verify the test delivers the expected result.

Our tests are also written using the filter pattern:

Project

Goals

The primary goals for Shoop, in no particular order:

An oft cited criticism of PHP is its inconsistent API. PHP's creator, Rasmus, has explained it this way (paraphrased):

PHP is perfectly consistent, just not the way you expect. It's vertically consistent. So, for every function in PHP, if you look at what's underneath it, the libc function under some of the string functions, for example, the argument order and naming matches what they're built upon. So, there's not consistency horitzontally, but there's perfect consistency vertically digging down into the stack.

If you use classes from the Illuminate Support portion of Laravel or some of the Symfony Components, you're familiar with the desire for horizontally consistent APIs problem (even beyond PHP itself).

While this immplementation is language-specific, the fundamental concepts, patterns, and naming strive to be language agnostic.

PHP is extremely simple for new developers, Shoop follows this tradition.

Contributing

For more general advice see our Contributing documentation.

What's in a name?

Shoop, as an acronym, points to the insipirations of the library:

Shoop, as a word, is akin to "photoshopping" (and sounds nicer than "Foops").

Shoop, as a name, is the title of a song by Salt-N-Pepa released in 1993 and used in the first installment of the Deadpool franchise in 2016.

Other

History

This library has been under development since the beginning of 2019 and has been used in the majority of 8fold projects since the middle of 2019. With every new project created we tried to go without it but found ourselves becoming annoyed, which is why we've decided to make it a more formal project and library consumable by others.


We offer multiple Interfaces and default Implementations for juggling between the supported, concrete types. Each interface offers two methods: one returns an object implementing the interface and the other returns the PHP types. The former are prefixed with "as" for use in Fluent Interfaces. The latter are prefixed with "ef" and can be thought of as similar to PHP Magic Methods which are prefixed with a double-underscore (__).

?? - abstract and concrete filters - ??

PHP deviations

Boolean

Shoop Shoop result PHP equivalent PHP result
TypeAsInteger::apply()->unfoldUsing(false) 0 count(false) PHP warning
TypeAsString::apply()->unfoldUsing(false) "false" (string) false "0"
TypeAsArray::apply()->unfoldUsing(false) [0 => true, 1 => false] (array) false [0] => false
TypeAsDictionary::apply()->unfoldUsing(false) ["false" => true, "true" => false] '' ''
TypeAsTuple::apply()->unfoldUsing(false) object(["false"] => true, ["true"] => false) (object) false object(["scalar"] => false)

Number and integer

Shoop Shoop result PHP equivalent PHP result
TypeAsInteger::apply()->unfoldUsing(2) 2 count(2) PHP warning
TypeAsArray::apply()->unfoldUsing(2) [0 => 0, 1 => 1, 2 => 2] (array) 2 [0 => 2]
TypeAsDictionary::apply()->unfoldUsing(2) ["i0" => 0, "i1" => 1, "i2" => 2] '' ''
TypeAsTuple::apply()->unfoldUsing(2) object(["i0"] => 0, ["i1"] => 1, ["i2"] => 2]) (object) 2 object(["scalar"] => 2)

Should array to tuple be the PHP default for array to object?? Reduces deviations.

String

Shoop Shoop result PHP equivalent PHP result
TypeAsInteger::apply()->unfoldUsing("Hi!") 3 (int) "Hi!" 0
TypeAsInteger::apply()->unfoldUsing("Hi!") 3 count("Hi!") PHP warning
TypeAsArray::apply()->unfoldUsing("Hi!") [0 => "H", 1 => "i", 2 => "!"] (array) "Hi!" [0 => "Hi!"]
TypeAsDictionary::apply()->unfoldUsing("Hi!") ["content" => "Hi!"] '' ''
TypeAsTuple::apply()->unfoldUsing("Hi!") object(["content"] => "Hi!") (object) "Hi!" object(["scalar"] => "Hi!")

Dictionary, tuple, and JSON

Ditionary and tuple deviate from PHP in similar ways, syntax might be different.

Shoop Shoop result PHP equivalent PHP result
TypeAsInteger::apply()->unfoldUsing(["a" => 1, "b" => 2]) 2 (int) ["a" => 1, "b" => 2] 1
TypeAsInteger::apply()->unfoldUsing(["a" => 1, "b" => 2]) 2 count(["a" => 1, "b" => 2]) 2
TypeAsString::apply()->unfoldUsing(["a" => 1, "b" => 2]) "", configurable (string) ["a" => 1, "b" => 2] PHP Notice: Array to string...
TypeAsArray::apply()->unfoldUsing(["a" => 1, "b" => 2]) [0 => 1, 1 => 2] (array) ["a" => 1, "b" => 2] ["a" => 1, "b" => 2]

Note: A JSON string is converted to a Tuple, and a Tuple is converted to a Dictionary.

Note: The default implementation of the PHP JsonSerialize interface results in the PHP type being converted to a Shoop Tuple, then being encoded.

Array

Shoop Shoop result PHP equivalent PHP result
TypeAsInteger::apply()->unfoldUsing(["a", "b"]) 2 (int) ["a", "b"] 1
TypeAsString::apply()->unfoldUsing(["a", "b"]) "ab", configurable (string) ["a", "b"] PHP Notice: Array to string...
TypeAsDictionary::apply()->unfoldUsing(["a", "b"]) ["i0" => "a", "i1" => "b"] (array) ["a", "b"] ["a", "b"]
TypeAsTuple::apply()->unfoldUsing(["a", "b"]) object(["i0"] => "a", ["i1"] => "b") (object) ["a", "b"] object(["0"] => "a", ["1"] => "b")

Should array to tuple be the PHP default for array to object?? Reduces deviations. Accessing those properties doesn't work as expected.

ex. $object = object(["0"] => 1, ["1"] => 2):

Object

ex. $using = new class {}

Shoop Shoop result PHP equivalent PHP result
TypeAsBoolean::apply()->unfoldUsing($using) false: Opposite of IsEmpty, can be overridden (bool) $using true, cannot be overridden
IsEmpty::apply()->unfoldUsing($using) true: Boolean of TypeAsInteger, can be overridden empty($using) false, cannot be overridden
TypeAsInteger::apply()->unfoldUsing($using) |0 (count of public properties), can be overridden |(int) $using` PHP Notice...
TypeAsNumber::apply()->unfoldUsing($using) |0.0 (count of public properties) |(float) $using` ''
TypeAsString::apply()->unfoldUsing($using) |"" (concatenated string properties), can be overridden |(string) $using` ''
TypeAsArray::apply()->unfoldUsing($using) |[] |(array) $using` []
TypeAsTuple::apply()->unfoldUsing($using) |object(): all methods and private properties removed |(object) $using` object(): all methods are removed, private properties remain

All versions of php-shoop with dependencies

PHP Build Version
Package Version
Requires php Version ~7.4|^8.0
8fold/php-foldable Version ~1.3.1
nikic/iter Version ^2.0
Composer command for our command line client (download client) This client runs in each environment. You don't need a specific PHP version etc. The first 20 API calls are free. Standard composer command

The package 8fold/php-shoop contains the following files

Loading the files please wait ....