diff --git a/.github/workflows/linux.yml b/.github/workflows/linux.yml
new file mode 100644
index 00000000..d451f017
--- /dev/null
+++ b/.github/workflows/linux.yml
@@ -0,0 +1,79 @@
+name: CI (Linux)
+
+on: [push, pull_request]
+
+jobs:
+ build_and_test:
+ strategy:
+ fail-fast: false
+ matrix:
+ version:
+ - 1.42.0 # MSRV
+ - stable
+ - nightly
+
+ name: ${{ matrix.version }} - x86_64-unknown-linux-gnu
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@master
+
+ - name: Install ${{ matrix.version }}
+ uses: actions-rs/toolchain@v1
+ with:
+ toolchain: ${{ matrix.version }}-x86_64-unknown-linux-gnu
+ profile: minimal
+ override: true
+
+ - name: Generate Cargo.lock
+ uses: actions-rs/cargo@v1
+ with:
+ command: generate-lockfile
+
+ - name: Cache cargo registry
+ uses: actions/cache@v1
+ with:
+ path: ~/.cargo/registry
+ key: ${{ matrix.version }}-x86_64-unknown-linux-gnu-cargo-registry-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: Cache cargo index
+ uses: actions/cache@v1
+ with:
+ path: ~/.cargo/git
+ key: ${{ matrix.version }}-x86_64-unknown-linux-gnu-cargo-index-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: Cache cargo build
+ uses: actions/cache@v1
+ with:
+ path: target
+ key: ${{ matrix.version }}-x86_64-unknown-linux-gnu-cargo-build-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: check build
+ uses: actions-rs/cargo@v1
+ with:
+ command: check
+ args: --all --all-features --bins --examples --tests
+
+ - name: tests
+ uses: actions-rs/cargo@v1
+ timeout-minutes: 40
+ with:
+ command: test
+ args: --all --all-features --no-fail-fast -- --nocapture
+
+ - name: Generate coverage file
+ if: matrix.version == 'stable' && (github.ref == 'refs/heads/master' || github.event_name == 'pull_request')
+ run: |
+ cargo install cargo-tarpaulin
+ cargo tarpaulin --out Xml --all --all-features
+
+ - name: Upload to Codecov
+ if: matrix.version == 'stable' && (github.ref == 'refs/heads/master' || github.event_name == 'pull_request')
+ uses: codecov/codecov-action@v1
+ with:
+ file: cobertura.xml
+
+ - name: Clear the cargo caches
+ run: |
+ cargo install cargo-cache --no-default-features --features ci-autoclean
+ cargo-cache
diff --git a/.github/workflows/osx.yml b/.github/workflows/osx.yml
new file mode 100644
index 00000000..cb3ad867
--- /dev/null
+++ b/.github/workflows/osx.yml
@@ -0,0 +1,65 @@
+name: CI (OSX)
+
+on: [push, pull_request]
+
+jobs:
+ build_and_test:
+ strategy:
+ fail-fast: false
+ matrix:
+ version:
+ - stable
+ - nightly
+
+ name: ${{ matrix.version }} - x86_64-apple-darwin
+ runs-on: macOS-latest
+
+ steps:
+ - uses: actions/checkout@master
+
+ - name: Install ${{ matrix.version }}
+ uses: actions-rs/toolchain@v1
+ with:
+ toolchain: ${{ matrix.version }}-x86_64-apple-darwin
+ profile: minimal
+ override: true
+
+ - name: Generate Cargo.lock
+ uses: actions-rs/cargo@v1
+ with:
+ command: generate-lockfile
+
+ - name: Cache cargo registry
+ uses: actions/cache@v1
+ with:
+ path: ~/.cargo/registry
+ key: ${{ matrix.version }}-x86_64-apple-darwin-cargo-registry-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: Cache cargo index
+ uses: actions/cache@v1
+ with:
+ path: ~/.cargo/git
+ key: ${{ matrix.version }}-x86_64-apple-darwin-cargo-index-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: Cache cargo build
+ uses: actions/cache@v1
+ with:
+ path: target
+ key: ${{ matrix.version }}-x86_64-apple-darwin-cargo-build-trimmed-${{ hashFiles('**/Cargo.lock') }}
+
+ - name: check build
+ uses: actions-rs/cargo@v1
+ with:
+ command: check
+ args: --all --all-features --bins --examples --tests
+
+ - name: tests
+ uses: actions-rs/cargo@v1
+ with:
+ command: test
+ args: --all --all-features --no-fail-fast -- --nocapture
+
+ - name: Clear the cargo caches
+ run: |
+ cargo install cargo-cache --no-default-features --features ci-autoclean
+ cargo-cache
diff --git a/.github/workflows/windows.yml b/.github/workflows/windows.yml
new file mode 100644
index 00000000..7c5c4287
--- /dev/null
+++ b/.github/workflows/windows.yml
@@ -0,0 +1,49 @@
+name: CI (Windows)
+
+on: [push, pull_request]
+
+env:
+ VCPKGRS_DYNAMIC: 1
+
+jobs:
+ build_and_test:
+ strategy:
+ fail-fast: false
+ matrix:
+ version:
+ - stable
+ - nightly
+
+ name: ${{ matrix.version }} - x86_64-pc-windows-msvc
+ runs-on: windows-latest
+
+ steps:
+ - uses: actions/checkout@master
+
+ - name: Install ${{ matrix.version }}
+ uses: actions-rs/toolchain@v1
+ with:
+ toolchain: ${{ matrix.version }}-x86_64-pc-windows-msvc
+ profile: minimal
+ override: true
+
+ - name: Install OpenSSL
+ run: |
+ vcpkg integrate install
+ vcpkg install openssl:x64-windows
+ Copy-Item C:\vcpkg\installed\x64-windows\bin\libcrypto-1_1-x64.dll C:\vcpkg\installed\x64-windows\bin\libcrypto.dll
+ Copy-Item C:\vcpkg\installed\x64-windows\bin\libssl-1_1-x64.dll C:\vcpkg\installed\x64-windows\bin\libssl.dll
+ Get-ChildItem C:\vcpkg\installed\x64-windows\bin
+ Get-ChildItem C:\vcpkg\installed\x64-windows\lib
+
+ - name: check build
+ uses: actions-rs/cargo@v1
+ with:
+ command: check
+ args: --all --all-features --bins --examples --tests
+
+ - name: tests
+ uses: actions-rs/cargo@v1
+ with:
+ command: test
+ args: --all --all-features --no-fail-fast -- --nocapture
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 00000000..93b7b2d1
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,13 @@
+/target
+Cargo.lock
+guide/build/
+/gh-pages
+
+*.so
+*.out
+*.pid
+*.sock
+*~
+
+# These are backup files generated by rustfmt
+**/*.rs.bk
diff --git a/Cargo.toml b/Cargo.toml
new file mode 100644
index 00000000..99f84fc3
--- /dev/null
+++ b/Cargo.toml
@@ -0,0 +1,4 @@
+[workspace]
+members = [
+ "ntex-multipart",
+]
\ No newline at end of file
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 00000000..0eed26a7
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,25 @@
+Copyright (c) 2020 Nikolay Kim, krircc
+
+Permission is hereby granted, free of charge, to any
+person obtaining a copy of this software and associated
+documentation files (the "Software"), to deal in the
+Software without restriction, including without
+limitation the rights to use, copy, modify, merge,
+publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software
+is furnished to do so, subject to the following
+conditions:
+
+The above copyright notice and this permission notice
+shall be included in all copies or substantial portions
+of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
+ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
+TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
+PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
+SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
+IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/README.md b/README.md
index 6c2130df..515a1c89 100644
--- a/README.md
+++ b/README.md
@@ -1,2 +1,26 @@
-ntex-extras
-===========
\ No newline at end of file
+
+
ntex-extras
+
A collection of additional crates supporting the ntex frameworks.
+
+
+[/badge.svg)](https://travis-ci.org/ntex-rs/ntex)
+[](https://blog.rust-lang.org/2020/03/12/Rust-1.42.html)
+
+
+
+
+
+## Crates
+
+| Crate | | |
+| -------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------- |
+| [ntex-multipart] | [](https://crates.io/crates/ntex-multipart) [](https://docs.rs/ntex-multipart) | Multipart support for ntex applications. |
+
+
+[ntex-multipart]: ntex-multipart
+
+## License
+
+This project is licensed under
+
+* MIT license ([LICENSE-MIT](LICENSE-MIT) or [http://opensource.org/licenses/MIT](http://opensource.org/licenses/MIT))
diff --git a/ntex-multipart/CHANGES.md b/ntex-multipart/CHANGES.md
new file mode 100644
index 00000000..b203b14e
--- /dev/null
+++ b/ntex-multipart/CHANGES.md
@@ -0,0 +1,5 @@
+# Changes
+
+## [0.1.0] - 2020-04-05
+
+* Fork to ntex namespace
diff --git a/ntex-multipart/Cargo.toml b/ntex-multipart/Cargo.toml
new file mode 100644
index 00000000..60d58118
--- /dev/null
+++ b/ntex-multipart/Cargo.toml
@@ -0,0 +1,25 @@
+[package]
+name = "ntex-multipart"
+version = "0.1.0"
+authors = ["krircc "]
+description = "Multipart support for ntex framework."
+readme = "README.md"
+keywords = ["http", "web", "framework", "async", "futures"]
+homepage = "https://ntex.rs"
+repository = "https://github.com/ntex/ntex-extras.git"
+documentation = "https://docs.rs/ntex-multipart/"
+license = "MIT"
+edition = "2018"
+
+[dependencies]
+ntex = "0.1.1"
+bytes = "0.5.3"
+derive_more = "0.99.2"
+httparse = "1.3"
+futures = "0.3.1"
+log = "0.4"
+mime = "0.3"
+twoway = "0.2"
+
+[dev-dependencies]
+ntex = "0.1.1"
\ No newline at end of file
diff --git a/ntex-multipart/LICENSE b/ntex-multipart/LICENSE
new file mode 100644
index 00000000..a7f2f828
--- /dev/null
+++ b/ntex-multipart/LICENSE
@@ -0,0 +1,25 @@
+Copyright (c) 2020 krircc
+
+Permission is hereby granted, free of charge, to any
+person obtaining a copy of this software and associated
+documentation files (the "Software"), to deal in the
+Software without restriction, including without
+limitation the rights to use, copy, modify, merge,
+publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software
+is furnished to do so, subject to the following
+conditions:
+
+The above copyright notice and this permission notice
+shall be included in all copies or substantial portions
+of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
+ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
+TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
+PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
+SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
+IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/ntex-multipart/README.md b/ntex-multipart/README.md
new file mode 100644
index 00000000..f252f263
--- /dev/null
+++ b/ntex-multipart/README.md
@@ -0,0 +1,7 @@
+# Multipart support for ntex framework
+
+## Documentation & community resources
+
+* [API Documentation](https://docs.rs/ntex-multipart/)
+* Cargo package: [actix-multipart](https://crates.io/crates/ntex-multipart)
+* Minimum supported Rust version: 1.42 or later
diff --git a/ntex-multipart/src/error.rs b/ntex-multipart/src/error.rs
new file mode 100644
index 00000000..44b5d3d3
--- /dev/null
+++ b/ntex-multipart/src/error.rs
@@ -0,0 +1,55 @@
+//! Error and Result module
+use derive_more::{Display, From};
+use ntex::http::error::{ParseError, PayloadError};
+use ntex::http::StatusCode;
+use ntex::web::{DefaultError, WebResponseError};
+
+/// A set of errors that can occur during parsing multipart streams
+#[derive(Debug, Display, From)]
+pub enum MultipartError {
+ /// Content-Type header is not found
+ #[display(fmt = "No Content-type header found")]
+ NoContentType,
+ /// Can not parse Content-Type header
+ #[display(fmt = "Can not parse Content-Type header")]
+ ParseContentType,
+ /// Multipart boundary is not found
+ #[display(fmt = "Multipart boundary is not found")]
+ Boundary,
+ /// Nested multipart is not supported
+ #[display(fmt = "Nested multipart is not supported")]
+ Nested,
+ /// Multipart stream is incomplete
+ #[display(fmt = "Multipart stream is incomplete")]
+ Incomplete,
+ /// Error during field parsing
+ #[display(fmt = "{}", _0)]
+ Parse(ParseError),
+ /// Payload error
+ #[display(fmt = "{}", _0)]
+ Payload(PayloadError),
+ /// Not consumed
+ #[display(fmt = "Multipart stream is not consumed")]
+ NotConsumed,
+}
+
+impl std::error::Error for MultipartError {}
+
+/// Return `BadRequest` for `MultipartError`
+impl WebResponseError for MultipartError {
+ fn status_code(&self) -> StatusCode {
+ StatusCode::BAD_REQUEST
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::*;
+ use ntex::web::HttpResponse;
+
+ #[test]
+ fn test_multipart_error() {
+ let resp: HttpResponse = MultipartError::Boundary.error_response();
+ assert_eq!(resp.status(), StatusCode::BAD_REQUEST);
+ }
+}
diff --git a/ntex-multipart/src/extractor.rs b/ntex-multipart/src/extractor.rs
new file mode 100644
index 00000000..a64876c4
--- /dev/null
+++ b/ntex-multipart/src/extractor.rs
@@ -0,0 +1,40 @@
+use futures::future::{ok, Ready};
+use ntex::http::Payload;
+use ntex::web::{ErrorRenderer, FromRequest, HttpRequest};
+
+use crate::server::Multipart;
+
+/// Get request's payload as multipart stream
+///
+/// Content-type: multipart/form-data;
+///
+/// ## Server example
+///
+/// ```rust
+/// use futures::{Stream, StreamExt};
+/// use ntex::web::{HttpResponse, Error};
+/// use ntex_multipart as mp;
+///
+/// async fn index(mut payload: mp::Multipart) -> Result {
+/// // iterate over multipart stream
+/// while let Some(item) = payload.next().await {
+/// let mut field = item?;
+///
+/// // Field in turn is stream of *Bytes* object
+/// while let Some(chunk) = field.next().await {
+/// println!("-- CHUNK: \n{:?}", std::str::from_utf8(&chunk?));
+/// }
+/// }
+/// Ok(HttpResponse::Ok().into())
+/// }
+/// # fn main() {}
+/// ```
+impl FromRequest for Multipart {
+ type Error = Err::Container;
+ type Future = Ready>;
+
+ #[inline]
+ fn from_request(req: &HttpRequest, payload: &mut Payload) -> Self::Future {
+ ok(Multipart::new(req.headers(), payload.take()))
+ }
+}
diff --git a/ntex-multipart/src/lib.rs b/ntex-multipart/src/lib.rs
new file mode 100644
index 00000000..43eb048c
--- /dev/null
+++ b/ntex-multipart/src/lib.rs
@@ -0,0 +1,8 @@
+#![allow(clippy::borrow_interior_mutable_const)]
+
+mod error;
+mod extractor;
+mod server;
+
+pub use self::error::MultipartError;
+pub use self::server::{Field, Multipart};
diff --git a/ntex-multipart/src/server.rs b/ntex-multipart/src/server.rs
new file mode 100644
index 00000000..6b2011fa
--- /dev/null
+++ b/ntex-multipart/src/server.rs
@@ -0,0 +1,793 @@
+//! Multipart payload support
+use std::cell::{Cell, RefCell, RefMut};
+use std::convert::TryFrom;
+use std::marker::PhantomData;
+use std::pin::Pin;
+use std::rc::Rc;
+use std::task::{Context, Poll};
+use std::{cmp, fmt};
+
+use bytes::{Bytes, BytesMut};
+use futures::stream::{LocalBoxStream, Stream, StreamExt};
+use httparse;
+use mime;
+
+use ntex::http::error::{ParseError, PayloadError};
+use ntex::http::header::{self, HeaderMap, HeaderName, HeaderValue};
+use ntex::task::LocalWaker;
+
+use crate::error::MultipartError;
+
+const MAX_HEADERS: usize = 32;
+
+/// The server-side implementation of `multipart/form-data` requests.
+///
+/// This will parse the incoming stream into `MultipartItem` instances via its
+/// Stream implementation.
+/// `MultipartItem::Field` contains multipart field. `MultipartItem::Multipart`
+/// is used for nested multipart streams.
+pub struct Multipart {
+ safety: Safety,
+ error: Option,
+ inner: Option>>,
+}
+
+enum InnerMultipartItem {
+ None,
+ Field(Rc>),
+}
+
+#[derive(PartialEq, Debug)]
+enum InnerState {
+ /// Stream eof
+ Eof,
+ /// Skip data until first boundary
+ FirstBoundary,
+ /// Reading boundary
+ Boundary,
+ /// Reading Headers,
+ Headers,
+}
+
+struct InnerMultipart {
+ payload: PayloadRef,
+ boundary: String,
+ state: InnerState,
+ item: InnerMultipartItem,
+}
+
+impl Multipart {
+ /// Create multipart instance for boundary.
+ pub fn new(headers: &HeaderMap, stream: S) -> Multipart
+ where
+ S: Stream- > + Unpin + 'static,
+ {
+ match Self::boundary(headers) {
+ Ok(boundary) => Multipart {
+ error: None,
+ safety: Safety::new(),
+ inner: Some(Rc::new(RefCell::new(InnerMultipart {
+ boundary,
+ payload: PayloadRef::new(PayloadBuffer::new(Box::new(stream))),
+ state: InnerState::FirstBoundary,
+ item: InnerMultipartItem::None,
+ }))),
+ },
+ Err(err) => Multipart {
+ error: Some(err),
+ safety: Safety::new(),
+ inner: None,
+ },
+ }
+ }
+
+ /// Extract boundary info from headers.
+ fn boundary(headers: &HeaderMap) -> Result {
+ if let Some(content_type) = headers.get(&header::CONTENT_TYPE) {
+ if let Ok(content_type) = content_type.to_str() {
+ if let Ok(ct) = content_type.parse::() {
+ if let Some(boundary) = ct.get_param(mime::BOUNDARY) {
+ Ok(boundary.as_str().to_owned())
+ } else {
+ Err(MultipartError::Boundary)
+ }
+ } else {
+ Err(MultipartError::ParseContentType)
+ }
+ } else {
+ Err(MultipartError::ParseContentType)
+ }
+ } else {
+ Err(MultipartError::NoContentType)
+ }
+ }
+}
+
+impl Stream for Multipart {
+ type Item = Result;
+
+ fn poll_next(
+ mut self: Pin<&mut Self>,
+ cx: &mut Context,
+ ) -> Poll