• Stars
    star
    132
  • Rank 274,205 (Top 6 %)
  • Language
    PHP
  • License
    MIT License
  • Created almost 9 years ago
  • Updated 5 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

PSR-7 and PSR-15 CORS middleware

PSR-7 and PSR-15 CORS Middleware

This middleware implements Cross-origin resource sharing. It supports both PSR-7 style doublepass and PSR-15 middleware standards. It has been tested with Slim Framework and Zend Expressive. Internally the middleware uses neomerx/cors-psr7 library for heavy lifting.

Latest Version Packagist Software License Build Status Coverage

Install

Install using composer.

$ composer require tuupola/cors-middleware

Usage

Documentation assumes you have working knowledge of CORS. There are no mandatory parameters. If you are using Zend Expressive skeleton middlewares are added to file called config/pipeline.php. Note that you must disable the default ImplicitOptionsMiddleware for this middleware to work.

use Tuupola\Middleware\CorsMiddleware;

#$app->pipe(ImplicitOptionsMiddleware::class);
$app->pipe(CorsMiddleware::class);

Slim Framework does not have specified config files. Otherwise adding the middleware is similar with previous.

$app->add(new Tuupola\Middleware\CorsMiddleware);

Rest of the examples use Slim Framework.

If called without any parameters the following defaults are used.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "origin" => ["*"],
    "methods" => ["GET", "POST", "PUT", "PATCH", "DELETE"],
    "headers.allow" => [],
    "headers.expose" => [],
    "credentials" => false,
    "cache" => 0,
]));
$ curl "https://api.example.com/" \
    --request OPTIONS \
    --include
    --header "Access-Control-Request-Method: PUT" \
    --header "Origin: http://www.example.com"

HTTP/1.1 200 OK
Access-Control-Allow-Origin: http://www.example.com
Vary: Origin
Access-Control-Allow-Methods: GET, POST, PUT, PATCH, DELETE

However, you most likely want to change some of the defaults. For example if developing a REST API which supports caching and conditional requests you could use the following.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "origin" => ["*"],
    "methods" => ["GET", "POST", "PUT", "PATCH", "DELETE"],
    "headers.allow" => ["Authorization", "If-Match", "If-Unmodified-Since"],
    "headers.expose" => ["Etag"],
    "credentials" => true,
    "cache" => 86400
]));
$ curl "https://api.example.com/foo" \
    --request OPTIONS \
    --include \
    --header "Origin: http://www.example.com" \
    --header "Access-Control-Request-Method: PUT" \
    --header "Access-Control-Request-Headers: Authorization, If-Match"

HTTP/1.1 200 OK
Access-Control-Allow-Origin: http://www.example.com
Access-Control-Allow-Credentials: true
Vary: Origin
Access-Control-Max-Age: 86400
Access-Control-Allow-Methods: GET, POST, PUT, PATCH, DELETE
Access-Control-Allow-Headers: authorization, if-match, if-unmodified-since
$ curl "https://api.example.com/foo" \
    --request PUT \
    --include \
    --header "Origin: http://www.example.com"

HTTP/1.1 200 OK
Access-Control-Allow-Origin: http://www.example.com
Access-Control-Allow-Credentials: true
Vary: Origin
Access-Control-Expose-Headers: Etag

Parameters

Origin

By default all origins are allowed. You can limit allowed origins by passing them as an array.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "origin" => ["app-1.example.com", "app-2.example.com"]
]));

You can also use wildcards to define multiple origins at once. Wildcards are matched by using the fnmatch() function.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "origin" => ["*.example.com"]
]));

Methods

Methods can be passed either as an array or a callable which returns an array. Below example is for Zend Expressive where value of methods is dynamic depending on the requested route.

use Tuupola\Middleware\CorsMiddleware;
use Zend\Expressive\Router\RouteResult;

$app->pipe(new CorsMiddleware([
    "origin" => ["*"],
    "methods" => function($request) {
        $result = $request->getAttribute(RouteResult::class);
        $route = $result->getMatchedRoute();
        return $route->getAllowedMethods();
    }
]));

Same thing for Slim 3. This assumes you have not defined the OPTIONS route.

use Fastroute\Dispatcher;
use Tuupola\Middleware\CorsMiddleware;

$app->add(
    new CorsMiddleware([
        "origin" => ["*"],
        "methods" => function($request) use ($app) {
            $container = $app->getContainer();
            $dispatch = $container["router"]->dispatch($request);
            if (Dispatcher::METHOD_NOT_ALLOWED === $dispatch[0]) {
                return $dispatch[1];
            }
        }
    ])
);

Logger

The optional logger parameter allows you to pass in a PSR-3 compatible logger to help with debugging or other application logging needs.

$logger = Monolog\Logger("slim");
$rotating = new RotatingFileHandler(__DIR__ . "/logs/slim.log", 0, Logger::DEBUG);
$logger->pushHandler($rotating);

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "logger" => $logger,
]));

Error

Error is called when CORS request fails. It receives last error message in arguments. This can be used for example to create application/json responses when CORS request fails.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "methods" => ["GET", "POST", "PUT"],
    "error" => function ($request, $response, $arguments) {
        $data["status"] = "error";
        $data["message"] = $arguments["message"];
        return $response
            ->withHeader("Content-Type", "application/json")
            ->write(json_encode($data, JSON_UNESCAPED_SLASHES | JSON_PRETTY_PRINT));
    }
]));
$ curl https://api.example.com/foo \
    --request OPTIONS \
    --include \
    --header "Access-Control-Request-Method: PATCH" \
    --header "Origin: http://www.example.com"

HTTP/1.1 401 Unauthorized
Content-Type: application/json
Content-Length: 83

{
    "status": "error",
    "message": "CORS requested method is not supported."
}

Server origin

If your same-origin requests contain an unnecessary Origin header, they might get blocked in case the server origin is not among the allowed origins already. In this case you can use the optional origin.server parameter to specify the origin of the server.

$app->add(new Tuupola\Middleware\CorsMiddleware([
    "origin.server" => "https://example.com"
]));
$ curl https://example.com/api \
    --request POST \
    --include \
    --header "Origin: https://example.com"

HTTP/1.1 200 OK

Testing

You can run tests either manually or automatically on every code change. Automatic tests require entr to work.

$ make test
$ brew install entr
$ make watch

Contributing

Please see CONTRIBUTING for details.

Security

If you discover any security related issues, please email [email protected] instead of using the issue tracker.

License

The MIT License (MIT). Please see License File for more information.

More Repositories

1

lazyload

Vanilla JavaScript plugin for lazyloading images
JavaScript
8,761
star
2

slim-jwt-auth

PSR-7 and PSR-15 JWT Authentication Middleware
PHP
828
star
3

jquery_chained

Chained Selects for jQuery and Zepto
JavaScript
589
star
4

slim-basic-auth

PSR-7 and PSR-15 HTTP Basic Authentication Middleware
PHP
440
star
5

slim-api-skeleton

Slim 3 API skeleton project for Composer
PHP
312
star
6

hagl

Hardware Agnostic Graphics Library for embedded
C
309
star
7

jquery_viewport

Add viewport selectors to jQuery. For example $("img:below-the-fold").something()
HTML
256
star
8

branca-spec

Authenticated and encrypted API tokens using modern crypto
223
star
9

server-timing-middleware

PSR-7 & PSR-15 middleware to add the Server-Timing header
PHP
198
star
10

base62

Base62 encoder and decoder for arbitrary data
PHP
195
star
11

micropython-mpu9250

MicroPython I2C driver for MPU9250 9-axis motion tracking device
Python
146
star
12

ksuid

K-Sortable Globally Unique IDs for PHP
PHP
103
star
13

branca-js

Authenticated encrypted API Tokens for JavaScript.
JavaScript
94
star
14

php_google_maps

API to help working with Google Static Maps.
PHP
84
star
15

micropython-m5stack

MicroPython Kitchen Sink for M5Stack
Python
81
star
16

avr_demo

Atmel demo code. Does not use any Arduino libraries. I want to learn this the hard way (tm).
Makefile
80
star
17

slim-image-resize

Image Resize Middleware for Slim Framework
PHP
60
star
18

branca-php

Authenticated and encrypted API tokens using modern crypto
PHP
52
star
19

base58

Base58 encoder and decoder for arbitrary data
PHP
51
star
20

esp_video

Proof of concept video player for ESP32 boards
C
48
star
21

hagl_esp_mipi

ESP32 MIPI DCS abstraction layer for the HAGL graphics library
C
48
star
22

pybranca

Authenticated Encrypted API Tokens for Python.
Python
47
star
23

instrument

PHP application instrumentation toolkit
PHP
44
star
24

hagl_pico_mipi

Raspberry Pi Pico MIPI DCS absraction layer for the HAGL graphics library
C
44
star
25

esp_effects

Old school demo effects for ESP32
C
42
star
26

whereami

Common PHP interface for wifi positioning services
PHP
34
star
27

slim-facebook

Boilerplate Facebook application with Slim, Eloquent and Facebook API.
JavaScript
33
star
28

demo_code

Miscallenous demo code.
JavaScript
32
star
29

axp192

Platform agnostic I2C driver for AXP192 power system management IC
C
31
star
30

esp_examples

The mandatory ESP-IDF (ESP32) examples repository
C
30
star
31

jquery_filestyle

jQuery plugin for styling file input elements.
JavaScript
28
star
32

base85

Base85 encoder and decoder for arbitrary data
PHP
27
star
33

jquery_dictionary

Spotlight searchable offline jQuery API documentation for OS X.
25
star
34

trilateration

Trilateration for PHP
PHP
24
star
35

pico_effects

Old school demo effects for Raspberry Pi Pico
C
23
star
36

pcf8563

Platform agnostic I2C driver for PCF8563 RTC
C
22
star
37

esp_software_i2c

Software I2C driver for ESP-IDF (ESP32)
C
22
star
38

witchcraft

Opionated PHP magic methods as traits for PHP 5.4+
PHP
21
star
39

micropython-mpu6886

MicroPython I2C driver for MPU6886 6-axis motion tracking device
Python
20
star
40

micropython-ili934x

MicroPython SPI Driver for ILI934X Series Based TFT / LCD Displays
Python
18
star
41

http-factory

Lightweight autodiscovering PSR-17 HTTP factories
PHP
17
star
42

wolf_assets

Mephisto style assets management for Wolf CMS. This plugin is currently unmaintained. Let me know if you want to take over.
PHP
17
star
43

php-docker-k8s

Examples how to run PHP with Docker and Kubernetes
PHP
16
star
44

flat-ui-theme

Flat UI theme for TextMate 2 and Sublime Text
15
star
45

sdl2_effects

Old school demo effects with HAGL and SDL2
C
14
star
46

embedded-fonts

FONTX fonts for embedded projects
C
14
star
47

base32

Base32 encoder and decoder for arbitrary data
PHP
13
star
48

instrument-middleware

PSR-7 Middleware for instrumenting PHP applications
PHP
11
star
49

slim-todo-backend

Slim 3 + Spot example for todobackend.com
PHP
11
star
50

slim-skeleton

Slim 3 + Datamapper + Monolog + Plates project for Composer
PHP
11
star
51

callable-handler

Compatibility layer between PSR-7 double pass and PSR-15 middlewares
PHP
10
star
52

bm8563

Platform agnostic I2C driver for BM8563 RTC
C
9
star
53

hagl_sdl2

SDL2 abstraction layer for the HAGL graphics library
C
9
star
54

branca-middleware

PSR-7 and PSR-15 Branca token authentication middleware
PHP
9
star
55

micropython-examples

The mandatory various experiments with MicroPython repository
Python
9
star
56

micropython-lis2hh12

MicroPython I2C driver for LIS2HH12 3-axis accelerometer
Python
9
star
57

wolf_dashboard

Provides simple admin dashboard to Wolf CMS.
PHP
9
star
58

branca-elixir

Secure alternative to JWT. Authenticated encrypted API tokens for Elixir.
Elixir
8
star
59

frog_jquery

Add jQuery to Frog CMS admin interface.
PHP
8
star
60

jquery_jeditable_wysiwyg

Wysiwyg input for Jeditable.
JavaScript
8
star
61

wolf_funky_cache

Funky cache plugin for Wolf CMS
PHP
7
star
62

beeper

Generic paginator for PHP 7.1+
PHP
7
star
63

jquery_jeditable_markitup

Universal markup editor for Jeditable.
7
star
64

axp173

Platform agnostic I2C driver for AXP173 power system management IC
C
6
star
65

triple-a

Alternative AVR (Arduino) Library
C
6
star
66

esp_gfx

HAGL speed tests for ESP32 boards
C
6
star
67

axp202

Platform agnostic I2C driver for AXP202 PMU
C
6
star
68

wolf_first_child

Redirect to first child page behaviour for Wolf CMS.
PHP
6
star
69

php_record

Simple Active Record implementation in PHP.
PHP
6
star
70

esp_twatch2020

Kitchen sink project for T-Watch 2020 and ESP-IDF
C
6
star
71

micropython-gnssl76l

MicroPython I2C driver for Quectel GNSS L76-L (GPS)
Python
6
star
72

rack-facebook-method-fix

Rack middleware to fix Facebook always POST problem.
Ruby
6
star
73

rack-funky-cache

Funky caching for Rack based applications
Ruby
5
star
74

frog_email_template

Provides form mailer backend to Frog CMS. This code is no longer maintained. Please go to: https://github.com/dajare/wolf_email_template
PHP
5
star
75

mephisto_sitemap

Mephisto plugin which implements (Google, MSN, Yahoo ) Sitemap protocol.
Ruby
5
star
76

trytes

Trytes encoder and decoder for arbitrary data
PHP
4
star
77

wolf_jquery_ui

Add jQuery UI to Wolf CMS admin interface.
PHP
4
star
78

em-websocket-server

Simple websocket server with EventMachine
Ruby
4
star
79

esp_i2c_helper

ESP I2C master HAL for hardware agnostic I2C drivers
C
4
star
80

screenshot

Full size website screenshots with Sinatra.
JavaScript
4
star
81

pico_gfx

HAGL speed tests for Rasrpberry Pi Pico boards
C
4
star
82

hagl_esp_solomon

ESP32 Solomon HAL for the HAGL graphics library
C
4
star
83

esp_mipi

Low level MIPI DCS compatible display driver for ESP-IDF
C
4
star
84

hagl_gd

GD HAL for the HAGL graphics library
C
3
star
85

fontx_tools

C
3
star
86

gd32v_effects

Old school demo effects for GD32V
C
3
star
87

slim-fake-mcrypt

Fake PHP mcrypt extension to enable installing Slim 2.3.* with Composer
3
star
88

dbal-psr3-logger

PSR-3 Logger for Doctrine DBAL
PHP
3
star
89

ulid

Universally Unique Lexicographically Sortable Identifier
PHP
3
star
90

corelocation

PHP implementation of Apple location services protocol
PHP
3
star
91

rack-lazy-load

Companion Rack middleware for Lazy Load jQuery plugin.
Ruby
2
star
92

hagl_gd32v_mipi

GD32V MIPI DCS HAL for the HAGL graphics library
C
2
star
93

esp_m5stick

Kitchen sink project for M5StickC and ESP-IDF
C
2
star
94

appelsiini.net

Sources and build files for my personal blog.
HTML
2
star
95

madgwick-backup

Backup of Sebastian Madgwick's original LGPL licensed sensor fusion algorithm
C
2
star
96

branca-cli

Command line tool for encoding, decoding and inspecting Branca tokens
JavaScript
2
star
97

wolf_jquery_tools

Add jQuery Tools to Wolf CMS admin interface.
PHP
2
star
98

arduino_xylophone

Awesome christmas xylophone done with Arduino and Sinatra.
JavaScript
1
star
99

messente

An object-oriented Ruby wrapper for the Messente API.
Ruby
1
star
100

gd32v_gfx

HAGL speed tests for GD32V boards
C
1
star