Spec names are intended to have enduring global semantics. So the notion of 
the same spec name having different semantics at different times seems to 
be at odds with that.

In general, it's often helpful to think about all the possible values that 
an attribute will have - that's the true spec. In this case, it seems like 
what you have for ::coercible-job-type is the true spec - it can be either 
a string or other named value.

And then think about the places where you need to add additional 
constraints to your ::job-record spec to narrow it. Like maybe here:

(s/fdef ... :ret (s/and ::job-record #(-> % ::job-type string?)))

You're on the brink of suggesting subtypes and covariant / contravariants, 
etc and that's a path spec is not going down. 


On Tuesday, February 7, 2017 at 9:40:24 AM UTC-6, Dave Tenny wrote:
>
> Let's say I have these definitions for a "job" record I'm managing, 
> perhaps in a database.
>
> (s/def ::job-status #{:in-progress :completed})
> (s/def ::user-id (s/and integer? #(>= % 0)))
> (s/def ::job-id integer?)
>
> (s/def ::coercible-job-type (s/and named? #(not (empty? (name %)))))
> (s/def ::job-type (s/and string? #(not (empty? %))))
>
> So we have a job status which can be any of a limited set of keywords,
> integer user and job ID's.  
>
> Then there's job type and my issue.  Depending on the context I want 
> either to insist that job type will be purely a non-empty string, or that 
> it may also be a 'named?' object, 
> i.e. something for which the clojure.core/name function will work.
>
> If I'm just returning a pure job record, my definition would look like 
> this:
>
> (s/def ::job-record 
>   (s/keys :req-un [::job-id ::user-id ::job-type ::job-status]))
>
> So a map with all those keys and type definitions. And that's great.  I 
> define functions that return these
> maps and so there's an (s/fdef ... :ret ::job-record).
>
> Now let's say I have an update! function that can take a map of optional 
> fields and update the record in the database with the fields that were 
> specified.
>
> (s/def ::field-options (s/keys :opt-un [::job-id ::user-id ::job-type 
> ::job-status]))
> (s/fdef update! :args (s/cat ::field-map ::field-options) ...)
> (defn update! [field-map] ... (:job-type field-map) ...)
>
> So far, so good. I have a parameter 'field-map' that can take optional map 
> keys, validate map arguments if the testing instrumentation is enabled, etc.
>
> Here is my problem.  There appears to be no way for me to define, in this 
> namespace, map specs with a :job-type key, but that will have 
> :coercible-job-type semantics,
> for my update! function. Or at least I don't see an easy way short of 
> writing a function to do everything that the existing machinery is doing 
> with respect to key and value validation,
> which is a lot of work to handle this tiny thing, basically the ability to 
> have map keys and spec semantics bound to different names in s/keys specs.
>
> For example, the following pseudo-spec might let me define a map key 
> :job-type whose semantics were specified by the ::coercible-job-type 
> definition.
> Or inlining of definitions which the spec guide says was omitted on 
> purpose.
>
> (s/def ::field-map (s/keys :opt-un [... [::coercible-job-type :as ::job-
> type]]))
>
> I want the map key to be ":job-type", but the clojure.spec semantics for 
> that field, *in selective module definitions*, to have ::coercible-job-type 
> behavior.
>
> So... am I missing something?
>
>

-- 
You received this message because you are subscribed to the Google
Groups "Clojure" group.
To post to this group, send email to clojure@googlegroups.com
Note that posts from new members are moderated - please be patient with your 
first post.
To unsubscribe from this group, send email to
clojure+unsubscr...@googlegroups.com
For more options, visit this group at
http://groups.google.com/group/clojure?hl=en
--- 
You received this message because you are subscribed to the Google Groups 
"Clojure" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to clojure+unsubscr...@googlegroups.com.
For more options, visit https://groups.google.com/d/optout.

Reply via email to