monad-oxide 0.12.0 → 0.13.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0be7d626dfbb5f8398d3b3ac58c4228bc28323fc0d64d74c4cd6d7e4f5966698
4
- data.tar.gz: fed978afd4d3ea108a8ab1fece39379573ade7bab1ae167c92ea9a6ca8bb3836
3
+ metadata.gz: f39d28705c0ba38e6cdaff1b0dfbd12468f7f8529d16d51b89b9583d5ddc484d
4
+ data.tar.gz: 245f18cb7377cfe3ff4ecee26d553d1bd695daa01ba0919a73218520ce23e980
5
5
  SHA512:
6
- metadata.gz: 90a2f66f1e4bcd4f08a39c96aa7c9b2d65dd7cdc5c92a9e7f3cb7660f00d64bae5645250fcb25fc9b4688b8d7bece8e4ceaa470de634ac868db03b4a025f1871
7
- data.tar.gz: 9b64b9ad89870629f44a593429e1f43b70ee04124e139ce334db65acf805ccb436c88992846eee80d09baf1c97a6f60bdf660382aa318b1042b05c742a3757c3
6
+ metadata.gz: a0c170fddf3095ffa876701fdaa799b5385055098e583ad1e82400c183204a9b5801558ead2dd265c8f9fe5eaca42c0e100ee15f9f76adab8e2e7f309fea5aa9
7
+ data.tar.gz: 4c82cfb3ae795077dfb14b62e3e9c663edd44cca6fd0b35b24e23d84f4b3fef53f02a0439e668eb0856c2956608e5fe150fe2ab562178e97b843f8aa1139763a
data/lib/either.rb ADDED
@@ -0,0 +1,73 @@
1
+ # frozen_string_literal: true
2
+ ################################################################################
3
+ # Provide an Either type (either one thing or another thing). Either represents
4
+ # this with a Left or a Right. The notable difference is Either doesn't assume
5
+ # one of the branches must be some kind of error - in fact no branch of Either
6
+ # has any preference to Left or Right.
7
+ #
8
+ # This is not part of Rust's standard library but it fits well with the monadic
9
+ # operations that Rust has with Result and Option. This particular
10
+ # implementation is a port from the Either library for Rust
11
+ # (https://docs.rs/either/latest/either/enum.Either.html).
12
+ #
13
+ # Be mindful that an Either can raise exceptions if errors are present in its
14
+ # calls. Since Either is not strictly compatible with Result, we cannot safely
15
+ # convert some operation to a Result implicitly.
16
+ ################################################################################
17
+
18
+ require_relative './error'
19
+
20
+ module MonadOxide
21
+
22
+ ##
23
+ # Thie Exception signals an area under construction, or somehow the consumer
24
+ # wound up with a `Result' and not one of its subclasses (@see Ok and @see
25
+ # Err). Implementors of new methods on `Ok' and `Err' should create methods on
26
+ # `Result' as well which immediately raise this `Exception'.
27
+ class EitherMethodNotImplementedError < MonadOxideError; end
28
+
29
+ ##
30
+ # This `Exception' is produced when a method that expects the function or
31
+ # block to provide a `Result' but was given something else. Generally this
32
+ # Exception is not raised, but instead converts the Result into a an Err.
33
+ # Example methods with this behavior are Result#and_then and Result#or_else.
34
+ class EitherReturnExpectedError < MonadOxideError
35
+ ##
36
+ # The transformation expected a `Result' but got something else.
37
+ # @param data [Object] Whatever we got that wasn't a `Result'.
38
+ def initialize(data)
39
+ super("An Either was expected but got #{data.inspect()}.")
40
+ data = @data
41
+ end
42
+ attr_reader(:data)
43
+ end
44
+
45
+ class Either
46
+
47
+ def initialize(_data)
48
+ raise NoMethodError.new('Do not use Either directly. See Left and Right.')
49
+ end
50
+
51
+ ##
52
+ # Use pattern matching to work with both Left and Right variants. This is
53
+ # useful when it is desirable to have both variants handled in the same
54
+ # location. It can also be useful when either variant can coerced into a
55
+ # non-Result type.
56
+ #
57
+ # Ruby has no built-in pattern matching, but the next best thing is a Hash
58
+ # using the Either classes themselves as the keys.
59
+ #
60
+ # Tests for this are found in the tests of the Left and Right classes.
61
+ #
62
+ # @param matcher [Hash<Class, Proc<T | E, R>] matcher The matcher to match
63
+ # upon.
64
+ # @option matcher [Proc] MonadOxide::Left The branch to execute for Left.
65
+ # @option matcher [Proc] MonadOxide::Right The branch to execute for Right.
66
+ # @return [R] The return value of the executed Proc.
67
+ def match(matcher)
68
+ matcher[self.class()].call(@data)
69
+ end
70
+
71
+ end
72
+
73
+ end
data/lib/error.rb ADDED
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+ ################################################################################
3
+ # Houses generic exception types.
4
+ #
5
+ # Not to be mistaken with err.rb for MonadOxide::Err.
6
+ ################################################################################
7
+ module MonadOxide
8
+
9
+ ##
10
+ # All errors in monad-oxide should inherit from this error. This makes it easy
11
+ # to handle library-wide errors on the consumer side.
12
+ class MonadOxideError < StandardError; end
13
+
14
+ ##
15
+ # This `Exception' is raised when the consumer makes a dangerous wager
16
+ # about which state the `Result' is in, and lost. More specifically: An `Ok'
17
+ # cannot unwrap an Exception, and an `Err' cannot unwrap the `Ok' data.
18
+ class UnwrapError < MonadOxideError; end
19
+
20
+ end
data/lib/left.rb ADDED
@@ -0,0 +1,149 @@
1
+ # frozen_string_literal: true
2
+ ################################################################################
3
+ #
4
+ ################################################################################
5
+
6
+ require_relative './error'
7
+
8
+ module MonadOxide
9
+
10
+ ##
11
+ # `Left' represents an arbitrary branch of Either. Since Either is supposed
12
+ # to be unbiased, you can assign any bias you like to Left, though in some
13
+ # ecosystems (such as Haskell, where this Either is ultimately inspired from),
14
+ # the Left side is the less common/favorable one, but this is only out of
15
+ # convention.
16
+ class Left < Either
17
+
18
+ ##
19
+ # Constructs a `Left' with the data provided.
20
+ # @param data [Object] The inner data this Either encapsulates.
21
+ def initialize(data)
22
+ @data = data
23
+ end
24
+
25
+ ##
26
+ # Applies `f` or the block over the data and returns the same `Left`. No
27
+ # changes are applied. This is ideal for logging. @see Either#inspect_left
28
+ # for a higher level understanding how this works with Eithers in general.
29
+ # @param f [Proc<A, _>] The function to call. Could be a block instead.
30
+ # Takes an `A` the return is ignored.
31
+ # @yield Will yield a block that takes an A the return is ignored. Same as
32
+ # `f' parameter.
33
+ # @return [Either<A, B>] returns self.
34
+ def inspect_left(f=nil, &block)
35
+ (f || block).call(@data)
36
+ self
37
+ end
38
+
39
+ ##
40
+ # Falls through. @see Either#inspect_right for how this is handled in the
41
+ # Either case, and @see Right#inspect_right for how this is handled in the
42
+ # Right case.
43
+ # @param f [Proc] Optional Proc - ignored.
44
+ # @yield An ignored block.
45
+ # @return [Either] This Either.
46
+ def inspect_right(f=nil, &block)
47
+ self
48
+ end
49
+
50
+ ##
51
+ # Identifies that this is a `Left`.
52
+ # For counterparts:
53
+ # @see MonadOxide::Left#right?
54
+ # @see MonadOxide::Right#left?
55
+ # @see MonadOxide::Right#right?
56
+ # @return [Boolean] `true` because this is a `Left`.
57
+ def left?()
58
+ true
59
+ end
60
+
61
+ ##
62
+ # Invokes `f' or the block with the data and returns the Either returned
63
+ # from that. The return type is enforced.
64
+ # @param f [Proc<A, Result<C>>] The function to call. Could be a block
65
+ # instead. Takes an [A=Object] and must return a [Either<C, B>].
66
+ # @yield Will yield a block that takes an A and returns a Either<C>. Same as
67
+ # `f' parameter.
68
+ # @return [Left<C> | Right<C>] A new Result from `f' or the
69
+ # block. If the return value is a non-Either, this this will raise
70
+ # `EitherReturnExpectedError'.
71
+ def left_and_then(f=nil, &block)
72
+ either = (f || block).call(@data)
73
+ if either.is_a?(Either)
74
+ either
75
+ else
76
+ raise EitherReturnExpectedError.new(either)
77
+ end
78
+ end
79
+
80
+
81
+ ##
82
+ # Applies `f' or the block over the data and returns a new `Left' with the
83
+ # returned value.
84
+ # @param f [Proc<A, C>] The function to call. Could be a block
85
+ # instead. Takes an [A=Object] and returns a C.
86
+ # @yield Will yield a block that takes an A and returns a C. Same as
87
+ # `f' parameter.
88
+ # @return [Either<C, B>] A new `Left<C>' whose `C' is the return of `f' or
89
+ # the block.
90
+ def map_left(f=nil, &block)
91
+ Left.new((f || block).call(@data))
92
+ end
93
+
94
+ ##
95
+ # This is a no-op for Left. @see Right#map_right.
96
+ # @param f [Proc<A, C>] A dummy function. Not used.
97
+ # @yield A dummy block. Not used.
98
+ # @return [Either<A, B>] This `Left'.
99
+ def map_right(f=nil, &block)
100
+ self
101
+ end
102
+
103
+ ##
104
+ # Identifies that this is not a `Right`.
105
+ # For counterparts:
106
+ # @see MonadOxide::Left#left?
107
+ # @see MonadOxide::Right#left?
108
+ # @see MonadOxide::Right#right?
109
+ # @return [Boolean] `false` because this is not a `Right`.
110
+ def right?()
111
+ false
112
+ end
113
+
114
+ ##
115
+ # The Left equivalent to Right#left_and_then. This is a no-op for Left.
116
+ # @see Right#right_and_then.
117
+ # @param f [Proc<A, B>] A dummy function. Not used.
118
+ # @yield A dummy block. Not used.
119
+ # @return [Either<A, B>] This `Either'.
120
+ def right_and_then(f=nil, &block)
121
+ self
122
+ end
123
+
124
+ ##
125
+ # Dangerously access the `Left' data. If this is a `Right', an
126
+ # `UnwrapError` will be raised. It is recommended to use this for tests
127
+ # only.
128
+ # @return [A] The inner data of this `Left'.
129
+ def unwrap_left()
130
+ @data
131
+ end
132
+
133
+ ##
134
+ # Dangerously access the `Right' data. Since this is a `Left', it will
135
+ # always raise an error. @see Either#unwrap_right. It is recommended to
136
+ # use this for tests only.
137
+ # @return [E] The data of this `Right'.
138
+ def unwrap_right()
139
+ raise UnwrapError.new(
140
+ <<~EOE
141
+ #{self.class()} with #{@data.inspect()} could not be unwrapped as a
142
+ Right.
143
+ EOE
144
+ )
145
+ end
146
+
147
+ end
148
+
149
+ end
data/lib/monad-oxide.rb CHANGED
@@ -1,6 +1,9 @@
1
1
  require_relative './result'
2
2
  require_relative './err'
3
3
  require_relative './ok'
4
+ require_relative './either'
5
+ require_relative './left'
6
+ require_relative './right'
4
7
  require_relative './array'
5
8
  require_relative './version'
6
9
 
@@ -18,6 +21,10 @@ module MonadOxide
18
21
  MonadOxide::Err.new(data)
19
22
  end
20
23
 
24
+ def left(data)
25
+ MonadOxide::Left.new(data)
26
+ end
27
+
21
28
  ##
22
29
  # Create an `Ok' as a conveniece method.
23
30
  # @param data [Object] The inner data for this `Ok'.
@@ -25,4 +32,9 @@ module MonadOxide
25
32
  def ok(data)
26
33
  MonadOxide::Ok.new(data)
27
34
  end
35
+
36
+ def right(data)
37
+ MonadOxide::Right.new(data)
38
+ end
39
+
28
40
  end
data/lib/ok.rb CHANGED
@@ -1,4 +1,5 @@
1
1
  require_relative './result'
2
+ require_relative './error'
2
3
 
3
4
  module MonadOxide
4
5
  ##
@@ -53,7 +54,7 @@ module MonadOxide
53
54
 
54
55
  ##
55
56
  # Falls through. @see Result#inspect_err for how this is handled in either
56
- # Result case, and @see Err.inspect_err for how this is handled in the Err
57
+ # Result case, and @see Err#inspect_err for how this is handled in the Err
57
58
  # case.
58
59
  # @param f [Proc] Optional Proc - ignored.
59
60
  # @yield An ignored block.
@@ -81,7 +82,7 @@ module MonadOxide
81
82
  end
82
83
 
83
84
  ##
84
- # Applies `f' or the block over the data and returns a new new `Ok' with the
85
+ # Applies `f' or the block over the data and returns a new `Ok' with the
85
86
  # returned value.
86
87
  # @param f [Proc<A, B>] The function to call. Could be a block
87
88
  # instead. Takes an [A=Object] and returns a B.
data/lib/result.rb CHANGED
@@ -1,8 +1,20 @@
1
+ # frozen_string_literal: true
2
+ ################################################################################
3
+ # A monadic way of handling error based flow.
4
+ # A Result is a semi-abstract class, whose implementors are Ok or Err. Ok
5
+ # carries some data and represents a success case. Err carries an error and
6
+ # represents a failure case.
7
+
8
+ # Most methods on Result will follow a success or error "track". This means an
9
+ # Ok will execute all of the success methods, such as inspect_ok, map, and
10
+ # and_then. Err will execute error methods, such as inspect_err, map_err, and
11
+ # or_else. In cases where the method doesn't match the state of the Result
12
+ # (Ok/Err), they return self, which also means they just fall through.
13
+ ################################################################################
14
+
15
+ require_relative './error'
16
+
1
17
  module MonadOxide
2
- ##
3
- # All errors in monad-oxide should inherit from this error. This makes it easy
4
- # to handle library-wide errors on the consumer side.
5
- class MonadOxideError < StandardError; end
6
18
 
7
19
  ##
8
20
  # Thie Exception signals an area under construction, or somehow the consumer
@@ -27,12 +39,6 @@ module MonadOxide
27
39
  attr_reader(:data)
28
40
  end
29
41
 
30
- ##
31
- # This `Exception' is raised when the consumer makes a dangerous wager
32
- # about which state the `Result' is in, and lost. More specifically: An `Ok'
33
- # cannot unwrap an Exception, and an `Err' cannot unwrap the `Ok' data.
34
- class UnwrapError < MonadOxideError; end
35
-
36
42
  ##
37
43
  # A Result is a chainable series of sequential transformations. The Result
38
44
  # structure contains an `Ok<A> | Err<Exception>`. This is the central location
data/lib/right.rb ADDED
@@ -0,0 +1,146 @@
1
+ # frozen_string_literal: true
2
+ ################################################################################
3
+ #
4
+ ################################################################################
5
+
6
+ require_relative './error'
7
+
8
+ module MonadOxide
9
+
10
+ ##
11
+ # `Right' represents an arbitrary branch of Either. Since Either is supposed
12
+ # to be unbiased, you can assign any bias you like to Right, though in some
13
+ # ecosystems (such as Haskell, where this Either is ultimately inspired from),
14
+ # the Right side is the more common/favorable one, but this is only out of
15
+ # convention.
16
+ class Right < Either
17
+
18
+ def initialize(data)
19
+ @data = data
20
+ end
21
+
22
+ ##
23
+ # Falls through. @see Either#inspect_left for how this is handled in the
24
+ # Either case, and @see Left#inspect_left for how this is handled in the
25
+ # Left case.
26
+ # @param f [Proc] Optional Proc - ignored.
27
+ # @yield An ignored block.
28
+ # @return [Either] This Either.
29
+ def inspect_left(f=nil, &block)
30
+ self
31
+ end
32
+
33
+ ##
34
+ # Applies `f` or the block over the data and returns the same `Right`. No
35
+ # changes are applied. This is ideal for logging. @see
36
+ # Either#inspect_right for a higher level understanding how this works with
37
+ # Eithers in general.
38
+ # @param f [Proc<A, _>] The function to call. Could be a block instead.
39
+ # Takes an `A` the return is ignored.
40
+ # @yield Will yield a block that takes an A the return is ignored. Same as
41
+ # `f' parameter.
42
+ # @return [Either<A, B>] returns self.
43
+ def inspect_right(f=nil, &block)
44
+ (f || block).call(@data)
45
+ self
46
+ end
47
+
48
+ ##
49
+ # Identifies that this is not a `Left`.
50
+ # For counterparts:
51
+ # @see MonadOxide::Left#left?
52
+ # @see MonadOxide::Left#right??
53
+ # @see MonadOxide::Right#right?
54
+ # @return [Boolean] `false` because this is not a `Left`.
55
+ def left?()
56
+ false
57
+ end
58
+
59
+ ##
60
+ # The Right equivalent to Left#right_and_then. This is a no-op for Right.
61
+ # @see Lefft#right_and_then.
62
+ # @param f [Proc<A, B>] A dummy function. Not used.
63
+ # @yield A dummy block. Not used.
64
+ # @return [Either<A, B>] This `Either'.
65
+ def left_and_then(f=nil, &block)
66
+ self
67
+ end
68
+
69
+ ##
70
+ # This is a no-op for Right. @see Left#map_left.
71
+ # @param f [Proc<A, C>] A dummy function. Not used.
72
+ # @yield A dummy block. Not used.
73
+ # @return [Either<A, B>] This `Left'.
74
+ def map_left(f=nil, &block)
75
+ self
76
+ end
77
+
78
+ ##
79
+ # Applies `f' or the block over the data and returns a new `Right' with the
80
+ # returned value.
81
+ # @param f [Proc<A, C>] The function to call. Could be a block
82
+ # instead. Takes an [A=Object] and returns a C.
83
+ # @yield Will yield a block that takes an A and returns a C. Same as
84
+ # `f' parameter.
85
+ # @return [Either<A, C>] A new `Right<C>' whose `C' is the return of `f' or
86
+ # the block.
87
+ def map_right(f=nil, &block)
88
+ Right.new((f || block).call(@data))
89
+ end
90
+
91
+ ##
92
+ # Identifies that this is a `Right`.
93
+ # For counterparts:
94
+ # @see MonadOxide::Left#left?
95
+ # @see MonadOxide::Left#right?
96
+ # @see MonadOxide::Right#left?
97
+ # @return [Boolean] `true` because this is a `Right`.
98
+ def right?()
99
+ true
100
+ end
101
+
102
+ ##
103
+ # Invokes `f' or the block with the data and returns the Either returned
104
+ # from that. The return type is enforced.
105
+ # @param f [Proc<A, Result<C>>] The function to call. Could be a block
106
+ # instead. Takes an [A=Object] and must return a [Either<C, B>].
107
+ # @yield Will yield a block that takes an A and returns a Either<C>. Same as
108
+ # `f' parameter.
109
+ # @return [Left<C> | Right<C>] A new Result from `f' or the
110
+ # block. If the return value is a non-Either, this this will raise
111
+ # `EitherReturnExpectedError'.
112
+ def right_and_then(f=nil, &block)
113
+ either = (f || block).call(@data)
114
+ if either.is_a?(Either)
115
+ either
116
+ else
117
+ raise EitherReturnExpectedError.new(either)
118
+ end
119
+ end
120
+
121
+ ##
122
+ # Dangerously access the `Left' data. Since this is a `Right', it will
123
+ # always raise an error. @see Either#unwrap_left. It is recommended to use
124
+ # this for tests only.
125
+ # @return [A] The inner data of this `Right'.
126
+ def unwrap_left()
127
+ raise UnwrapError.new(
128
+ <<~EOE
129
+ #{self.class()} with #{@data.inspect()} could not be unwrapped as a
130
+ Left.
131
+ EOE
132
+ )
133
+ end
134
+
135
+ ##
136
+ # Dangerously access the `Right' data. If this is a `Left', an
137
+ # `UnwrapError` will be raised. It is recommended to use this for tests
138
+ # only.
139
+ # @return [A] The inner data of this `Right'.
140
+ def unwrap_right()
141
+ @data
142
+ end
143
+
144
+ end
145
+
146
+ end
data/lib/version.rb CHANGED
@@ -3,5 +3,5 @@ module MonadOxide
3
3
  # This version is locked to 0.x.0 because semver is a lie and this project
4
4
  # uses CICD to push new versions. Any commits that appear on main will result
5
5
  # in a new version of the gem created and published.
6
- VERSION='0.12.0'
6
+ VERSION='0.13.0'
7
7
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: monad-oxide
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.12.0
4
+ version: 0.13.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Logan Barnett
8
- autorequire:
8
+ autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2024-12-27 00:00:00.000000000 Z
11
+ date: 2025-03-06 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: org-ruby
@@ -70,23 +70,27 @@ description: |
70
70
  Monad-Oxide is a port of Rust's built-in monads from std, such as Result and
71
71
  Option. This enables better reasoning about error handling and possibly missing
72
72
  data.
73
- email:
73
+ email:
74
74
  executables: []
75
75
  extensions: []
76
76
  extra_rdoc_files: []
77
77
  files:
78
78
  - lib/array.rb
79
+ - lib/either.rb
79
80
  - lib/err.rb
81
+ - lib/error.rb
82
+ - lib/left.rb
80
83
  - lib/monad-oxide.rb
81
84
  - lib/ok.rb
82
85
  - lib/result.rb
86
+ - lib/right.rb
83
87
  - lib/version.rb
84
- homepage:
88
+ homepage:
85
89
  licenses:
86
90
  - Apache-2.0
87
91
  - MIT
88
92
  metadata: {}
89
- post_install_message:
93
+ post_install_message:
90
94
  rdoc_options: []
91
95
  require_paths:
92
96
  - lib
@@ -102,7 +106,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
102
106
  version: '0'
103
107
  requirements: []
104
108
  rubygems_version: 3.3.26
105
- signing_key:
109
+ signing_key:
106
110
  specification_version: 4
107
111
  summary: Ruby port of Rust's Result and Option.
108
112
  test_files: []