From ce543359634cbd2c4b19deedf62798157a9dd41d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Konrad=20Go=C5=82awski?= Date: Sun, 27 Oct 2024 17:56:18 +0100 Subject: [PATCH 1/2] Add async support for Tokio runtime --- .cargo/README.md | 2 +- CHANGELOG.md | 7 ++ Cargo.toml | 11 +- README.md | 2 +- src/lib.rs | 63 +++++++++- src/reader.rs | 28 +++-- src/writer.rs | 18 +++ tests/md5.rs | 322 +++++++++++++++++++++++++++++++++++++++++++++++ 8 files changed, 439 insertions(+), 14 deletions(-) diff --git a/.cargo/README.md b/.cargo/README.md index c47abe0..1345b3c 100644 --- a/.cargo/README.md +++ b/.cargo/README.md @@ -27,7 +27,7 @@ cargo add chksum-md5 ## Usage -Use the `chksum` function to calcualate digest of file, directory and so on. +Use the `chksum` function to calculate digest of file, directory and so on. ```rust use chksum_md5 as md5; diff --git a/CHANGELOG.md b/CHANGELOG.md index ac3b8ff..0cac446 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,10 +5,17 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [Unreleased] + +### Added + +- Added async support for Tokio runtime. + ## [0.0.0] - 2023-12-21 ### Added - Initial release. +[Unreleased]: https://github.com/chksum-rs/md5/compare/v0.0.0...HEAD [0.0.0]: https://github.com/chksum-rs/md5/releases/tag/v0.0.0 diff --git a/Cargo.toml b/Cargo.toml index 46296ad..4e15d56 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -16,16 +16,21 @@ all-features = true rustdoc-args = ["--cfg", "docsrs"] [dependencies] -chksum-core = "0.0.0" +chksum-core = "0.1.0" chksum-hash-md5 = "0.0.0" -chksum-reader = { version = "0.0.0", optional = true } -chksum-writer = { version = "0.0.0", optional = true } +chksum-reader = { version = "0.1.0", optional = true } +chksum-writer = { version = "0.1.0", optional = true } +tokio = { version = "1.37.0", features = ["io-util"], optional = true } [dev-dependencies] assert_fs = { version = "1.0.13", features = ["color-auto"] } thiserror = "1.0.51" +tokio = { version = "1.37.0", features = ["macros", "rt", "rt-multi-thread"] } [features] default = [] reader = ["chksum-reader"] writer = ["chksum-writer"] + +# async runtimes +async-runtime-tokio = ["chksum-core/async-runtime-tokio", "chksum-reader?/async-runtime-tokio", "chksum-writer?/async-runtime-tokio", "tokio"] diff --git a/README.md b/README.md index dfea589..eaa9854 100644 --- a/README.md +++ b/README.md @@ -27,7 +27,7 @@ cargo add chksum-md5 ## Usage -Use the `chksum` function to calcualate digest of file, directory and so on. +Use the `chksum` function to calculate digest of file, directory and so on. ```rust use chksum_md5 as md5; diff --git a/src/lib.rs b/src/lib.rs index de2f9cc..c1e5998 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -19,7 +19,7 @@ //! //! # Usage //! -//! Use the [`chksum`] function to calcualate digest of file, directory and so on. +//! Use the [`chksum`] function to calculate digest of file, directory and so on. //! //! ```rust //! # use std::path::Path; @@ -39,6 +39,30 @@ //! # } //! ``` //! +//! ## Asynchronous Runtime +//! +//! Use the [`async_chksum`] function to calculate digest of file, directory and so on. +//! +//! ```rust +//! # #[cfg(feature = "async-runtime-tokio")] +//! # { +//! # use std::path::Path; +//! # use chksum_md5::Result; +//! use chksum_md5 as md5; +//! use tokio::fs::File; +//! +//! # async fn wrapper(path: &Path) -> Result<()> { +//! let file = File::open(path).await?; +//! let digest = md5::async_chksum(file).await?; +//! assert_eq!( +//! digest.to_hex_lowercase(), +//! "5c71dbb287630d65ca93764c34d9aa0d" +//! ); +//! # Ok(()) +//! # } +//! # } +//! ``` +//! //! # Input Types //! //! ## Bytes @@ -231,6 +255,12 @@ //! cargo add chksum-md5 --features reader,writer //! ``` //! +//! ## Asynchronous Runtime +//! +//! * `async-runtime-tokio`: Enables async interface for Tokio runtime. +//! +//! By default, neither of these features is enabled. +//! //! # Disclaimer //! //! The MD5 hash function should be used only for backward compatibility due to security issues. @@ -252,14 +282,23 @@ pub mod writer; use std::fmt::{self, Display, Formatter, LowerHex, UpperHex}; use chksum_core as core; +#[cfg(feature = "async-runtime-tokio")] +#[doc(no_inline)] +pub use chksum_core::AsyncChksumable; #[doc(no_inline)] pub use chksum_core::{Chksumable, Error, Hash, Hashable, Result}; #[doc(no_inline)] pub use chksum_hash_md5 as hash; +#[cfg(all(feature = "reader", feature = "async-runtime-tokio"))] +#[doc(inline)] +pub use crate::reader::AsyncReader; #[cfg(feature = "reader")] #[doc(inline)] pub use crate::reader::Reader; +#[cfg(all(feature = "writer", feature = "async-runtime-tokio"))] +#[doc(inline)] +pub use crate::writer::AsyncWriter; #[cfg(feature = "writer")] #[doc(inline)] pub use crate::writer::Writer; @@ -341,6 +380,28 @@ pub fn chksum(data: impl core::Chksumable) -> Result { core::chksum::(data) } +/// Computes the hash of the given input. +/// +/// # Example +/// +/// ```rust +/// use chksum_md5 as md5; +/// +/// # async fn wrapper() { +/// let data = b"example data"; +/// if let Ok(digest) = md5::async_chksum(data).await { +/// assert_eq!( +/// digest.to_hex_lowercase(), +/// "5c71dbb287630d65ca93764c34d9aa0d" +/// ); +/// } +/// # } +/// ``` +#[cfg(feature = "async-runtime-tokio")] +pub async fn async_chksum(data: impl core::AsyncChksumable) -> Result { + core::async_chksum::(data).await +} + /// The MD5 hash instance. #[derive(Clone, Debug, Default, PartialEq, Eq)] pub struct MD5 { diff --git a/src/reader.rs b/src/reader.rs index 7ef21e2..8dcf7c7 100644 --- a/src/reader.rs +++ b/src/reader.rs @@ -47,24 +47,36 @@ use std::io::Read; use chksum_reader as reader; +#[cfg(feature = "async-runtime-tokio")] +use tokio::io::AsyncRead; use crate::MD5; /// A specialized [`Reader`](reader::Reader) type with the [`MD5`] hash algorithm. pub type Reader = reader::Reader; +#[cfg(feature = "async-runtime-tokio")] +/// A specialized [`AsyncReader`](reader::AsyncReader) type with the [`MD5`] hash algorithm. +pub type AsyncReader = reader::AsyncReader; + /// Creates new [`Reader`]. -pub fn new(inner: R) -> Reader -where - R: Read, -{ +pub fn new(inner: impl Read) -> Reader { reader::new(inner) } /// Creates new [`Reader`] with provided hash. -pub fn with_hash(inner: R, hash: MD5) -> Reader -where - R: Read, -{ +pub fn with_hash(inner: impl Read, hash: MD5) -> Reader { reader::with_hash(inner, hash) } + +#[cfg(feature = "async-runtime-tokio")] +/// Creates new [`AsyncReader`]. +pub fn async_new(inner: impl AsyncRead) -> AsyncReader { + reader::async_new(inner) +} + +#[cfg(feature = "async-runtime-tokio")] +/// Creates new [`AsyncReader`] with provided hash. +pub fn async_with_hash(inner: impl AsyncRead, hash: MD5) -> AsyncReader { + reader::async_with_hash(inner, hash) +} diff --git a/src/writer.rs b/src/writer.rs index 69a1057..27e0c29 100644 --- a/src/writer.rs +++ b/src/writer.rs @@ -45,12 +45,18 @@ use std::io::Write; use chksum_writer as writer; +#[cfg(feature = "async-runtime-tokio")] +use tokio::io::AsyncWrite; use crate::MD5; /// A specialized [`Writer`](writer::Writer) type with the [`MD5`] hash algorithm. pub type Writer = writer::Writer; +#[cfg(feature = "async-runtime-tokio")] +/// A specialized [`AsyncWriter`](writer::AsyncWriter) type with the [`MD5`] hash algorithm. +pub type AsyncWriter = writer::AsyncWriter; + /// Creates new [`Writer`]. pub fn new(inner: impl Write) -> Writer { writer::new(inner) @@ -60,3 +66,15 @@ pub fn new(inner: impl Write) -> Writer { pub fn with_hash(inner: impl Write, hash: MD5) -> Writer { writer::with_hash(inner, hash) } + +#[cfg(feature = "async-runtime-tokio")] +/// Creates new [`AsyncWriter`]. +pub fn async_new(inner: impl AsyncWrite) -> AsyncWriter { + writer::async_new(inner) +} + +#[cfg(feature = "async-runtime-tokio")] +/// Creates new [`AsyncWriter`] with provided hash. +pub fn async_with_hash(inner: impl AsyncWrite, hash: MD5) -> AsyncWriter { + writer::async_with_hash(inner, hash) +} diff --git a/tests/md5.rs b/tests/md5.rs index fa8fd26..dd65411 100644 --- a/tests/md5.rs +++ b/tests/md5.rs @@ -4,7 +4,11 @@ use std::io::Error as IoError; use assert_fs::fixture::FixtureError; use assert_fs::prelude::{FileTouch, FileWriteBin, PathChild}; use assert_fs::TempDir; +#[cfg(feature = "async-runtime-tokio")] +use chksum_md5::async_chksum; use chksum_md5::{chksum, Error as ChksumError}; +#[cfg(feature = "async-runtime-tokio")] +use tokio::fs::{read_dir as tokio_read_dir, File as TokioFile}; #[derive(Debug, thiserror::Error)] enum Error { @@ -27,6 +31,21 @@ fn empty_directory_as_path() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_directory_as_path() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + + let dir = temp_dir.path(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn empty_directory_as_pathbuf() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -42,6 +61,25 @@ fn empty_directory_as_pathbuf() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_directory_as_pathbuf() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + + let dir = temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + + let dir = &temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn empty_directory_as_readdir() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -53,6 +91,21 @@ fn empty_directory_as_readdir() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_directory_as_readdir() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + + let dir = tokio_read_dir(temp_dir.path()).await?; + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_empty_file_as_path() -> Result<(), Error> { let temp_dir = { @@ -68,6 +121,25 @@ fn non_empty_directory_with_empty_file_as_path() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_empty_file_as_path() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + temp_dir.child("file.txt").touch()?; + temp_dir + }; + + let dir = temp_dir.path(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_empty_file_as_pathbuf() -> Result<(), Error> { let temp_dir = { @@ -87,6 +159,29 @@ fn non_empty_directory_with_empty_file_as_pathbuf() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_empty_file_as_pathbuf() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + temp_dir.child("file.txt").touch()?; + temp_dir + }; + + let dir = temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + + let dir = &temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_empty_file_as_readdir() -> Result<(), Error> { let temp_dir = { @@ -102,6 +197,25 @@ fn non_empty_directory_with_empty_file_as_readdir() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_empty_file_as_readdir() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + temp_dir.child("file.txt").touch()?; + temp_dir + }; + + let dir = tokio_read_dir(temp_dir.path()).await?; + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_non_empty_file_as_path() -> Result<(), Error> { let temp_dir = { @@ -119,6 +233,27 @@ fn non_empty_directory_with_non_empty_file_as_path() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_non_empty_file_as_path() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + temp_dir + }; + + let dir = temp_dir.path(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_non_empty_file_as_pathbuf() -> Result<(), Error> { let temp_dir = { @@ -140,6 +275,31 @@ fn non_empty_directory_with_non_empty_file_as_pathbuf() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_non_empty_file_as_pathbuf() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + temp_dir + }; + + let dir = temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + + let dir = &temp_dir.to_path_buf(); + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} + #[test] fn non_empty_directory_with_non_empty_file_as_readdir() -> Result<(), Error> { let temp_dir = { @@ -157,6 +317,27 @@ fn non_empty_directory_with_non_empty_file_as_readdir() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_directory_with_non_empty_file_as_readdir() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = { + let temp_dir = TempDir::new()?; + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + temp_dir + }; + + let dir = tokio_read_dir(temp_dir.path()).await?; + let digest = async_chksum(dir).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} + #[test] fn empty_file_as_path() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -173,6 +354,26 @@ fn empty_file_as_path() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_file_as_path() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file + }; + + let file = child.path(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn empty_file_as_pathbuf() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -193,6 +394,30 @@ fn empty_file_as_pathbuf() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_file_as_pathbuf() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file + }; + + let file = child.to_path_buf(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + + let file = &child.to_path_buf(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn empty_file_as_file() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -213,6 +438,31 @@ fn empty_file_as_file() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_empty_file_as_file() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file + }; + + let file = TokioFile::open(child.path()).await?; + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + + // TODO: missing `&File` implementation + // let file = &TokioFile::open(child.path()).await?; + // let digest = async_chksum(file).await?.to_hex_lowercase(); + // assert_eq!(digest, "d41d8cd98f00b204e9800998ecf8427e"); + } + + Ok(()) +} + #[test] fn non_empty_file_as_path() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -230,6 +480,27 @@ fn non_empty_file_as_path() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_file_as_path() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + file + }; + + let file = child.path(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} + #[test] fn non_empty_file_as_pathbuf() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -251,6 +522,31 @@ fn non_empty_file_as_pathbuf() -> Result<(), Error> { Ok(()) } +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_file_as_pathbuf() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + file + }; + + let file = child.to_path_buf(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + + let file = &child.to_path_buf(); + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} + #[test] fn non_empty_file_as_file() -> Result<(), Error> { let temp_dir = TempDir::new()?; @@ -271,3 +567,29 @@ fn non_empty_file_as_file() -> Result<(), Error> { Ok(()) } + +#[cfg_attr(not(feature = "async-runtime-tokio"), ignore)] +#[tokio::test] +async fn async_runtime_tokio_non_empty_file_as_file() -> Result<(), Error> { + #[cfg(feature = "async-runtime-tokio")] + { + let temp_dir = TempDir::new()?; + let child = { + let file = temp_dir.child("file.txt"); + file.touch()?; + file.write_binary(b"data")?; + file + }; + + let file = TokioFile::open(child.path()).await?; + let digest = async_chksum(file).await?.to_hex_lowercase(); + assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + + // TODO: missing `&File` implementation + // let file = &TokioFile::open(child.path()).await?; + // let digest = async_chksum(file).await?.to_hex_lowercase(); + // assert_eq!(digest, "8d777f385d3dfec8815d20f7496026dc"); + } + + Ok(()) +} From 028836401caa350209ac6a3e197ab7588600dd34 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Konrad=20Go=C5=82awski?= Date: Sat, 7 Dec 2024 17:53:22 +0100 Subject: [PATCH 2/2] Update MSRV --- .cargo/README.md | 2 +- .github/workflows/rust.yml | 6 +++--- CHANGELOG.md | 4 ++++ Cargo.toml | 2 +- README.md | 2 +- 5 files changed, 10 insertions(+), 6 deletions(-) diff --git a/.cargo/README.md b/.cargo/README.md index 1345b3c..83268c4 100644 --- a/.cargo/README.md +++ b/.cargo/README.md @@ -3,7 +3,7 @@ [![GitHub](https://img.shields.io/badge/github-chksum--rs%2Fmd5-24292e?style=flat-square&logo=github "GitHub")](https://github.com/chksum-rs/md5) [![Build](https://img.shields.io/github/actions/workflow/status/chksum-rs/md5/rust.yml?branch=master&style=flat-square&logo=github "Build")](https://github.com/chksum-rs/md5/actions/workflows/rust.yml) [![docs.rs](https://img.shields.io/docsrs/chksum-md5?style=flat-square&logo=docsdotrs "docs.rs")](https://docs.rs/chksum-md5/) -[![MSRV](https://img.shields.io/badge/MSRV-1.70.0-informational?style=flat-square "MSRV")](https://github.com/chksum-rs/md5/blob/master/Cargo.toml) +[![MSRV](https://img.shields.io/badge/MSRV-1.74.0-informational?style=flat-square "MSRV")](https://github.com/chksum-rs/md5/blob/master/Cargo.toml) [![deps.rs](https://deps.rs/crate/chksum-md5/0.0.0/status.svg?style=flat-square "deps.rs")](https://deps.rs/crate/chksum-md5/0.0.0) [![unsafe forbidden](https://img.shields.io/badge/unsafe-forbidden-success.svg?style=flat-square "unsafe forbidden")](https://github.com/rust-secure-code/safety-dance) [![LICENSE](https://img.shields.io/github/license/chksum-rs/md5?style=flat-square "LICENSE")](https://github.com/chksum-rs/md5/blob/master/LICENSE) diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml index ecfab83..fab4591 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/rust.yml @@ -58,7 +58,7 @@ jobs: strategy: fail-fast: false matrix: - toolchain: [1.70.0, stable, nightly] + toolchain: [1.74.0, stable, nightly] name: "Build and test (OS: Linux, Toolchain: ${{ matrix.toolchain }})" steps: - name: Repository checkout @@ -87,7 +87,7 @@ jobs: strategy: fail-fast: false matrix: - toolchain: [1.70.0, stable, nightly] + toolchain: [1.74.0, stable, nightly] name: "Build and test (OS: MacOS, Toolchain: ${{ matrix.toolchain }})" steps: - name: Repository checkout @@ -116,7 +116,7 @@ jobs: strategy: fail-fast: false matrix: - toolchain: [1.70.0, stable, nightly] + toolchain: [1.74.0, stable, nightly] name: "Build and test (OS: Windows, Toolchain: ${{ matrix.toolchain }})" steps: - name: Repository checkout diff --git a/CHANGELOG.md b/CHANGELOG.md index 0cac446..6d8bdbc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added async support for Tokio runtime. +### Changed + +- Updated MSRV to `1.74.0`. + ## [0.0.0] - 2023-12-21 ### Added diff --git a/Cargo.toml b/Cargo.toml index 4e15d56..2222082 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -3,7 +3,7 @@ name = "chksum-md5" version = "0.0.0" authors = ["Konrad GoĊ‚awski "] edition = "2021" -rust-version = "1.70.0" +rust-version = "1.74.0" description = "An implementation of the MD5 hash function with a straightforward interface for computing digests of bytes, files, directories, and more." readme = ".cargo/README.md" repository = "https://github.com/chksum-rs/md5" diff --git a/README.md b/README.md index eaa9854..ed1b5bd 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ [![crates.io](https://img.shields.io/crates/v/chksum-md5?style=flat-square&logo=rust "crates.io")](https://crates.io/crates/chksum-md5) [![Build](https://img.shields.io/github/actions/workflow/status/chksum-rs/md5/rust.yml?branch=master&style=flat-square&logo=github "Build")](https://github.com/chksum-rs/md5/actions/workflows/rust.yml) [![docs.rs](https://img.shields.io/docsrs/chksum-md5?style=flat-square&logo=docsdotrs "docs.rs")](https://docs.rs/chksum-md5/) -[![MSRV](https://img.shields.io/badge/MSRV-1.70.0-informational?style=flat-square "MSRV")](https://github.com/chksum-rs/md5/blob/master/Cargo.toml) +[![MSRV](https://img.shields.io/badge/MSRV-1.74.0-informational?style=flat-square "MSRV")](https://github.com/chksum-rs/md5/blob/master/Cargo.toml) [![deps.rs](https://deps.rs/crate/chksum-md5/0.0.0/status.svg?style=flat-square "deps.rs")](https://deps.rs/crate/chksum-md5/0.0.0) [![unsafe forbidden](https://img.shields.io/badge/unsafe-forbidden-success.svg?style=flat-square "unsafe forbidden")](https://github.com/rust-secure-code/safety-dance) [![LICENSE](https://img.shields.io/github/license/chksum-rs/md5?style=flat-square "LICENSE")](https://github.com/chksum-rs/md5/blob/master/LICENSE)