Some time ago, I found a very powerful tool: Sourcery, which solves some of the problems I encountered in Swift development. Sourcery is not well known in the Chinese community, so I specially write an article to introduce it. This paper is roughly divided into three parts:
- Concepts and functions of metaprogramming
- Principle and basic use of Sourcery
- Sourcery and Codable practices
What is metaprogramming
The concept of meta-programming may be unfamiliar to many people, but partly because of translation problems, the word “meta” may seem rather confusing. In a word, metaprogramming is generating code from code.
This statement can be understood on two levels:
- Dynamically modify the structure of the program itself at run time through techniques such as reflection. Take, for example, the Objective-C Runtime, which we’re all familiar with.
- Specific code is generated through the DSL, usually during the compile time preprocessing phase.
OC has a very powerful Runtime feature, which allows you to view and modify all members of an object during Runtime, so there are JSON to Model libraries like Mantle. You can even add, remove, and replace methods in a type at run time, and of course you can add types dynamically, so Aspects and AOP come into play. All of these applications fall under the category of metaprogramming, because they do so by modifying the program itself at run time, a feature that saves us a lot of repetitive boilerplate code.
Swift is a statically strongly typed language and does not have Runtime features as powerful as OC. Although Swift can plug into OC Runtime, it is easy to turn your code into “OC written in Swift”, and changes to the Runtime can make the program difficult to understand. Swift provides a type named Mirror to check the properties of objects at runtime. However, Mirror can only be viewed and cannot be modified. On the other hand, it has poor performance and is recommended to be used only for debugging.
Therefore, the first way in Swift is not available, so we have to look at another way to find the answer. Fortunately, there is a mature solution, which is Sourcery.
Sourcery
Simply put, Sourcery is a Swift code generator that automatically generates Swift code based on our predefined templates.
The basic use
Define the template
Take the official Demo as an example. Let’s say you have a custom type:
struct Person {
var name: String
var age: Int
}
Copy the code
To implement the Equatable protocol for this type, you must compare the equality of each attribute in turn in the == method:
extension Person {
static func= =(lhs: Person, rhs: Person) -> Bool {
guard lhs.name == rhs.name else { return false }
guard lhs.age == rhs.age else { return false }
return true}}Copy the code
Typically, we have a large number of Model types in our projects, and implementing Equatable for all of them would involve a lot of repetitive work. And if you add new attributes to a type, you must change its Equatable implementation at the same time, otherwise unexpected bugs may occur.
Sourcery saves us from boilerplate code. First, we need to define a unified template for all Equatable implementations, which is written in a language called Stencil. Stencil is a Stencil language designed specifically for Swift. The syntax is very simple, and you can define templates like this for the above code (we recommend using VScode and Stencil for template writing) :
{% for type in types.implementing.AutoEquatable %}
extension {{type.name}}: Equatable {
static func= =(lhs: {{type.name}}, rhs: {{type.name}}) -> Bool{{%for variable in type.storedVariables %}
guard lhs.{{variable.name}} == rhs.{{variable.name}} else { return false }
{% endfor %}
return true
}
}
{% endfor %}
Copy the code
The AutoEquatable that appears in this code is a protocol predefined in our own code, just an empty protocol for markup:
protocol AutoEquatable {}Copy the code
Its role is to make what we need can be found in the template type, just put the Person type of custom declaration to achieve AutoEquatable, after can be used in the template types. The implementing. AutoEquatable find target type, Type. StoredVariables is then used to iterate over all the stored attributes in the type to generate the corresponding comparison code.
Code generation
Once the template is defined, you can use it to generate code by first installing Sourcery on your system: brew install Sourcery. Then run the following command:
sourcery \
--sources ./YourProject \
--templates ./YourTemplates \
--output ./YourProject/AutoGenerated.swift
Copy the code
–source specifies the root directory of the project, –templates specifies the directory where the template files are stored, and –output prints the generated code to the specified path. You can also customize the parameters by using a.sourcery.yml file on the command line, which I won’t describe here.
You can then see a code file named AutoGenerated. Swift in the project path, which contains the following content:
/ / Generated using Sourcery 0.12.0 - https://github.com/krzysztofzablocki/Sourcery
// DO NOT EDIT
extension Person {
static func= =(lhs: Person, rhs: Person) -> Bool {
guard lhs.name == rhs.name else { return false }
guard lhs.age == rhs.age else { return false }
return true}}Copy the code
The generated code file is required to participate in the compilation, remember to add it to the project.
Next, we can integrate the code generation step into the Xcode compilation process by adding a script to Build Phases (here I have added the Sourcery binary to the project directory as well) :
Note that this script must be added before Compile Sources, otherwise the newly generated code will not be compiled. After that, as long as our type implements AutoEquatable, the Build code will update automatically every time we add or remove attributes, eliminating the need for manual changes.
The above Equatable is just an example, see the template autoequatable.stencil for the complete version
The principle of
As you can see from the examples above, the key to Sourcery’s power is the ability to retrieve all types of information in our code when parsing templates, which gives us a great deal of freedom when writing templates. Sourcery uses two key techniques to achieve all this: Stencil and SourceKitten.
Stencil
Stencil, as mentioned earlier, is a template language designed for Swift. Stencil’s syntax is very simple, parsing only three syntax patterns:
{{... }}
: Variable syntax, parses the middle part as a variable (or an expression for a variable), and inserts the parsed value into the template as a result.{%... %}
: Tag syntax. Tags are used to represent syntax with special functions, such as judgmentsif
And the cycle offor
.{#... #}
: Comment syntax, which does not appear in parsed results.
In addition, a Filter, the concept of its syntax is like this: {{” font “| uppercase}}. Symbol | input variables on the left, the right is a Filter, the output the uppercase string. Filter is essentially a method with Any input and output. For example, upperCase would look like this:
registerFilter("uppercase".filter: uppercase) // Inject a Filter
func uppercase(_ value: Any?) -> Any? {
return stringify(value).uppercased()
}
Copy the code
Also, variables that are accessible during template parsing are injected into the Stencil environment at run time. Stencil is very extensible, and there is a library on Github called StencilSwiftKit that extends Stencil with a lot more convenient syntax.
SourceKit
Xcode’s handling of Swift and OC is a bit different. OC’s compiler is executed in the Xcode process, while Swift’s compiler is executed in a separate process, involving a collection of compilation tools called SourceKit, and the compiled results communicate with Xcode via XPC.
SourceKitten is an open source library that interacts with SourceKit and returns the syntactic structure of the code as JSON. With SourceKitten, Sourcery can take information about all types in the code and inject them as variables into Stencil’s context, so we can walk through the code with {{types}} in the template.
Practice in Codable
The Demo described below has been uploaded to my Github: AutoCodableDemo.
Codable is native JSON parsing support introduced by Swift4. Compared to third-party libraries like ObjectMapper, Codable automatically interprets properties in Models, making it easy to use if your data Model and JSON structure are exactly the same.
To do this, you must first define an enumeration for the type that implements the CodingKey. This enumeration must contain all the property fields, even if the property does not need to be customized. To do more complex customizations, you need to implement your own init(From decoder: decoder) and encode(to Encoder: encoder) methods, and decode and encode operations for all attributes.
This code is obviously very repeatable and is well suited for automatic generation using Sourcery:
AutoCodable
First define an AutoCodable type in your project:
protocol AutoCodable: Codable {}Copy the code
Find all types that are AutoCodable in the template and automatically add an enumeration for them in the extension that contains all the attribute names:
enum CodingKeys: String.CodingKey {{%for var in type.storedVariables %}
case {{var.name}} {% if var|annotated:"key"%} ="{{var.annotations.key}}"{% endif %}
{% endfor %}
}
Copy the code
Sourcery provides a mechanism called annotation, which provides necessary data to templates in code in the form of annotations, by prefacing the definition of a variable or type with a line like this:
// sourcery: key = "value"
var something: Int
Copy the code
Sourcery parses the annotations in this format and adds key-value to the annotations attribute corresponding to that variable in the template, providing some custom data for template parsing in the code.
use
Make your custom types AutoCodable:
struct Person: AutoCodable {
var myName: String
}
Copy the code
AutoCodable provides the following features:
-
Custom field names: Prefix an annotation to an attribute that requires a custom field name:
// sourcery: key = "my_name" var myName: String Copy the code
-
Set a property default: AutoCodable allows you to provide a default value for a property in JSON instead of throwing an error if the field fails to be resolved. The property no longer needs to be defined as an optional type:
// sourcery: default = true var something: Bool Copy the code
-
Ignore a field: Ignored properties do not participate in Encode and Decode of JSON, and ignored properties must have a default value:
// sourcery: skip var something: Int = 0 Copy the code
-
Int resolver to Bool:
Codable has strict requirements for type when parsing JSON. If an attribute is a Bool, a type error will be thrown if the corresponding field value is an Int in JSON (unlike Mantle in OC, which automatically converts). While there’s nothing wrong with Codable, we already have a lot of background interface data that uses 1s and 0s for true and false in our actual projects. So AutoCodable does this for Bool types and supports resolving ints into Bool types.
Then add the generated Code file to the project as described above. You can see that Sourcery saves us a lot of Code repetition for custom parsing. The only drawback is that we can only comment the template by adding a Code Snippet: // Sourcery: <# Key #> = <#value#> can provide some help, as far as the name of the Key is concerned. On top of that, Sourcery has perfected my block on Codable.
conclusion
Sourcery is essentially a preprocessor, and it brings flexible metaprogramming features to Swift. You can even embed generated code in your own code, and its application scenarios go far beyond those described above. Programmers’ time is precious and we should focus on what’s really important. If you’re using Swift, try it out and say goodbye to boilerplate code that is trivial and repetitive .