Skip to content

Commit 6cffd04

Browse files
committed
Initial import
0 parents  commit 6cffd04

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

47 files changed

+3944
-0
lines changed

.github/ISSUE_TEMPLATE/BUG_REPORT.md

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
---
2+
name: 🐛 Bug Report
3+
about: Something isn't working as expected
4+
---
5+
6+
<!--
7+
Thanks for contributing to Swift Algorithms!
8+
9+
Before you submit your issue, please replace each paragraph
10+
below with the relevant details for your bug, and complete
11+
the steps in the checklist by placing an 'x' in each box:
12+
13+
- [x] I've completed this task
14+
- [ ] This task isn't completed
15+
-->
16+
17+
Replace this paragraph with a short description of the incorrect incorrect behavior. If this is a regression, please note the last version that the behavior was correct in addition to your current version.
18+
19+
**Swift Algorithms version:** `0.0.1` or the `main` branch, for example.
20+
**Swift version:** Paste the output of `swift --version` here.
21+
22+
### Checklist
23+
- [ ] If possible, I've reproduced the issue using the `main` branch of this package
24+
- [ ] I've searched for [existing GitHub issues](https://github.com/apple/swift-algorithms/issues)
25+
26+
### Steps to Reproduce
27+
Replace this paragraph with an explanation of how to reproduce the incorrect behavior. Include a simple code example, if possible.
28+
29+
### Expected behavior
30+
Describe what you expect to happen.
31+
32+
### Actual behavior
33+
Describe or copy/paste the behavior you observe.
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
---
2+
name: 💡 Feature Request
3+
about: A suggestion for a new feature
4+
---
5+
6+
<!--
7+
Thanks for contributing to Swift Algorithms!
8+
9+
Before you submit your issue, please replace the paragraph
10+
below with information about your proposed feature.
11+
-->
12+
13+
Replace this paragraph with a description of your proposed feature. Code samples that show what's missing, or what new capabilities will be possible, are very helpful! Provide links to existing issues or external references/discussions, if appropriate.

.github/ISSUE_TEMPLATE/config.yml

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
blank_issues_enabled: false
2+
contact_links:
3+
- name: ❓ Discussion Forum
4+
url: https://forums.swift.org/c/related-projects/algorithms/
5+
about: Questions about using Swift Algorithms? Ask here!

.github/PULL_REQUEST_TEMPLATE.md

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
<!--
2+
Thanks for contributing to Swift Algorithms!
3+
4+
If this pull request adds new API, please add '?template=new.md'
5+
to the URL to switch to the appropriate template.
6+
7+
Before you submit your request, please replace the paragraph
8+
below with the relevant details, and complete the steps in the
9+
checklist by placing an 'x' in each box:
10+
11+
- [x] I've completed this task
12+
- [ ] This task isn't completed
13+
-->
14+
15+
Replace this paragraph with a description of your changes and rationale. Provide links to an existing issue or external references/discussions, if appropriate.
16+
17+
### Checklist
18+
- [ ] I've added at least one test that validates that my change is working, if appropriate
19+
- [ ] I've followed the code style of the rest of the project
20+
- [ ] I've read the [Contribution Guidelines](CONTRIBUTING.md)
21+
- [ ] I've updated the documentation if necessary

.github/PULL_REQUEST_TEMPLATE/NEW.md

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
<!--
2+
Thanks for contributing to Swift Algorithms!
3+
4+
Before you submit your request, please replace each paragraph
5+
below with the relevant details, and complete the steps in the
6+
checklist by placing an 'x' in each box:
7+
8+
- [x] I've completed this task
9+
- [ ] This task isn't completed
10+
-->
11+
12+
### Description
13+
Replace this paragraph with a description of your changes and rationale. Provide links to an existing issue or external references/discussions, if appropriate.
14+
15+
### Detailed Design
16+
Include any additional information about the design here. At minimum, show any new API:
17+
18+
```swift
19+
extension Collection {
20+
/// The new feature implemented by this pull request.
21+
func newFeature()
22+
}
23+
```
24+
25+
### Documentation Plan
26+
How has the new feature been documented? Have the relevant portions of the guides been updated in addition to symbol-level documentation?
27+
28+
### Test Plan
29+
How is the new feature tested?
30+
31+
### Source Impact
32+
What is the impact of this change on existing users? Does it deprecate or remove any existing API?
33+
34+
### Checklist
35+
- [ ] I've added at least one test that validates that my change is working, if appropriate
36+
- [ ] I've followed the code style of the rest of the project
37+
- [ ] I've read the [Contribution Guidelines](CONTRIBUTING.md)
38+
- [ ] I've updated the documentation if necessary

.gitignore

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
.DS_Store
2+
/.build
3+
/Packages
4+
/*.xcodeproj
5+
xcuserdata/
6+
/.swiftpm

CHANGELOG.md

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# CHANGELOG
2+
3+
<!--
4+
Add new items at the end of the relevant section under **Unreleased**.
5+
-->
6+
7+
This project follows semantic versioning. While still in major version `0`,
8+
source-stability is only guaranteed within minor versions (e.g. between
9+
`0.0.3` and `0.0.4`). If you want to guard against potentially source-breaking
10+
package updates, you can specify your package dependency using
11+
`.upToNextMinor(from: "0.0.1")` as the requirement.
12+
13+
## [Unreleased]
14+
15+
*No changes yet.*
16+
17+
---
18+
19+
## [0.0.1] - 2020-10-07
20+
21+
- **Swift Algorithms** initial release.
22+
23+
---
24+
25+
This changelog's format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
26+
27+
<!-- Link references for releases -->
28+
29+
[Unreleased]: https://github.com/apple/swift-algorithms/compare/0.0.1...HEAD
30+
[0.0.1]: https://github.com/apple/swift-algorithms/releases/tag/0.0.1
31+
32+
<!-- Link references for pull requests -->
33+
34+
<!-- Link references for contributors -->
35+

CODE_OF_CONDUCT.md

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
# Code of Conduct
2+
To be a truly great community, Swift.org needs to welcome developers from all walks of life,
3+
with different backgrounds, and with a wide range of experience. A diverse and friendly
4+
community will have more great ideas, more unique perspectives, and produce more great
5+
code. We will work diligently to make the Swift community welcoming to everyone.
6+
7+
To give clarity of what is expected of our members, Swift.org has adopted the code of conduct
8+
defined by [contributor-covenant.org](https://www.contributor-covenant.org). This document is used across many open source
9+
communities, and we think it articulates our values well. The full text is copied below:
10+
11+
### Contributor Code of Conduct v1.3
12+
As contributors and maintainers of this project, and in the interest of fostering an open and
13+
welcoming community, we pledge to respect all people who contribute through reporting
14+
issues, posting feature requests, updating documentation, submitting pull requests or patches,
15+
and other activities.
16+
17+
We are committed to making participation in this project a harassment-free experience for
18+
everyone, regardless of level of experience, gender, gender identity and expression, sexual
19+
orientation, disability, personal appearance, body size, race, ethnicity, age, religion, or
20+
nationality.
21+
22+
Examples of unacceptable behavior by participants include:
23+
- The use of sexualized language or imagery
24+
- Personal attacks
25+
- Trolling or insulting/derogatory comments
26+
- Public or private harassment
27+
- Publishing other’s private information, such as physical or electronic addresses, without explicit permission
28+
- Other unethical or unprofessional conduct
29+
30+
Project maintainers have the right and responsibility to remove, edit, or reject comments,
31+
commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of
32+
Conduct, or to ban temporarily or permanently any contributor for other behaviors that they
33+
deem inappropriate, threatening, offensive, or harmful.
34+
35+
By adopting this Code of Conduct, project maintainers commit themselves to fairly and
36+
consistently applying these principles to every aspect of managing this project. Project
37+
maintainers who do not follow or enforce the Code of Conduct may be permanently removed
38+
from the project team.
39+
40+
This code of conduct applies both within project spaces and in public spaces when an
41+
individual is representing the project or its community.
42+
43+
Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by
44+
contacting a project maintainer at [[email protected]](mailto:[email protected]). All complaints will be reviewed and
45+
investigated and will result in a response that is deemed necessary and appropriate to the
46+
circumstances. Maintainers are obligated to maintain confidentiality with regard to the reporter
47+
of an incident.
48+
49+
*This policy is adapted from the Contributor Code of Conduct [version 1.3.0](http://contributor-covenant.org/version/1/3/0/).*
50+
51+
### Reporting
52+
A working group of community members is committed to promptly addressing any [reported
53+
issues](mailto:[email protected]). Working group members are volunteers appointed by the project lead, with a
54+
preference for individuals with varied backgrounds and perspectives. Membership is expected
55+
to change regularly, and may grow or shrink.

CONTRIBUTING.md

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
By submitting a pull request, you represent that you have the right to license
2+
your contribution to Apple and the community, and agree by submitting the patch
3+
that your contributions are licensed under the [Swift
4+
license](https://swift.org/LICENSE.txt).
5+
6+
---
7+
8+
Before submitting the pull request, please make sure you have tested your
9+
changes and that they follow the Swift project [guidelines for contributing
10+
code](https://swift.org/contributing/#contributing-code).

Guides/Chain.md

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
# Chain
2+
3+
[[Source](https://github.com/apple/swift-algorithms/blob/main/Sources/Algorithms/Chain.swift) |
4+
[Tests](https://github.com/apple/swift-algorithms/blob/main/Tests/AlgorithmsTests/ChainTests.swift)]
5+
6+
Concatenates two collections with the same element type, one after another.
7+
8+
This operation is available through the `chained(with:)` method on any sequence.
9+
10+
```swift
11+
let numbers = [10, 20, 30].chained(with: 1...5)
12+
// Array(numbers) == [10, 20, 30, 1, 2, 3, 4, 5]
13+
//
14+
let letters = "abcde".chained(with: "FGHIJ")
15+
// String(letters) == "abcdeFGHIJ"
16+
```
17+
18+
Unlike placing two collections in an array and calling `joined()`, chaining
19+
permits different collection types, performs no allocations, and can preserve
20+
the shared conformances of the two underlying types.
21+
22+
## Detailed Design
23+
24+
The `chained(with:)` method is added as an extension method on the `Sequence`
25+
protocol:
26+
27+
```swift
28+
extension Sequence {
29+
public func chained<S: Sequence>(with other: S) -> Concatenation<Self, S>
30+
where Element == S.Element
31+
}
32+
33+
```
34+
35+
The resulting `Chain` type is a sequence, with conditional conformance to the
36+
`Collection`, `BidirectionalCollection`, and `RandomAccessCollection` when both
37+
the first and second arguments conform.
38+
39+
### Naming
40+
41+
This method’s and type’s name match the term of art used in other languages and
42+
libraries.
43+
44+
### Comparison with other langauges
45+
46+
**Rust:** Rust provides a `chain` function that concatenates two iterators.
47+
48+
**Ruby/Python:** Ruby and Python’s `itertools` both define a `chain` function
49+
for concatenating collections of different kinds.

Guides/Chunked.md

+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
# Chunked
2+
3+
[[Source](https://github.com/apple/swift-algorithms/blob/main/Sources/Algorithms/Chunked.swift) |
4+
[Tests](https://github.com/apple/swift-algorithms/blob/main/Tests/AlgorithmsTests/ChunkedTests.swift)]
5+
6+
Break a collection into subsequences where consecutive elements pass a binary
7+
predicate, or where all elements in each chunk project to the same value.
8+
9+
There are two variations of the `chunked` method: `chunked(by:)` and
10+
`chunked(on:)`. `chunked(by:)` uses a binary predicate to test consecutive
11+
elements, separating chunks where the predicate returns `false`. For example,
12+
you can chunk a collection into ascending sequences using this method:
13+
14+
```swift
15+
let numbers = [10, 20, 30, 10, 40, 40, 10, 20]
16+
let chunks = numbers.chunked(by: { $0 <= $1 })
17+
// [[10, 20, 30], [10, 40, 40], [10, 20]]
18+
```
19+
20+
The `chunk(on:)` method, by contrast, takes a projection of each element and
21+
separates chunks where the projection of two consecutive elements is not equal.
22+
23+
```swift
24+
let names = ["David", "Kyle", "Karoy", "Nate"]
25+
let chunks = names.chunked(on: \.first!)
26+
// [["David"], ["Kyle", "Karoy"], ["Nate"]]
27+
```
28+
29+
These methods are related to the [existing SE proposal][proposal] for chunking a
30+
collection into subsequences of a particular size, potentially named something
31+
like `chunked(length:)`. Unlike the `split` family of methods, the entire
32+
collection is included in the chunked result — joining the resulting chunks
33+
recreates the original collection.
34+
35+
```swift
36+
c.elementsEqual(c.chunked(...).joined())
37+
// true
38+
```
39+
40+
[proposal]: https://github.com/apple/swift-evolution/pull/935
41+
42+
## Detailed Design
43+
44+
The two methods are added as extension to `Collection`, with two matching
45+
versions that return a lazy wrapper added to `LazyCollectionProtocol`.
46+
47+
```swift
48+
extension Collection {
49+
public func chunked(
50+
by belongInSameGroup: (Element, Element) -> Bool
51+
) -> [SubSequence]
52+
53+
public func chunked<Subject: Equatable>(
54+
on projection: (Element) -> Subject
55+
) -> [SubSequence]
56+
}
57+
58+
extension LazyCollectionProtocol {
59+
public func chunked(
60+
by belongInSameGroup: @escaping (Element, Element) -> Bool
61+
) -> LazyChunked<Elements>
62+
63+
public func chunked<Subject: Equatable>(
64+
on projection: @escaping (Element) -> Subject
65+
) -> LazyChunked<Elements>
66+
}
67+
```
68+
69+
The `LazyChunked` type is bidirectional when the wrapped collection is
70+
bidirectional.
71+
72+
### Complexity
73+
74+
The eager methods are O(_n_), the lazy methods are O(_1_).
75+
76+
### Naming
77+
78+
The operation performed by these methods is similar to other ways of breaking a collection up into subsequences. In particular, the predicate-based `split(where:)` method looks similar to `chunked(on:)`. You can draw a distinction between these different operations based on the resulting subsequences:
79+
80+
- `split`: *In the standard library.* Breaks a collection into subsequences, removing any elements that are considered "separators". The original collection cannot be recovered from the result of splitting.
81+
- `chunked`: *In this package.* Breaks a collection into subsequences, preserving each element in its initial ordering. Joining the resulting subsequences re-forms the original collection.
82+
- `sliced`: *Not included in this package or the stdlib.* Breaks a collection into potentially overlapping subsequences.
83+
84+
85+
### Comparison with other langauges
86+
87+
**Ruby:** Ruby’s `Enumerable` class defines `chunk_while` and `chunk`, which map
88+
to the proposed `chunked(by:)` and `chunked(on:)` methods.
89+
90+
**Rust:** Rust defines a variety of size-based `chunks` methods, but doesn’t
91+
include any with the functionality described here.

0 commit comments

Comments
 (0)