Xcode 10.2, macOS Mojave 10.14.4, iOS 12.1 и другие бета-версии

nh62fpv4v4whs_gzolx479bjnv0.jpeg

Вышли новые беты Xcode, macOS Mojave и iOS — и вот основные моменты, которые я усвоил.


Swift


Прежде всего, последняя бета-версия Xcode идет в комплекте с новой версией Swift:

Apple Swift 5.0 (swiftlang-1001.0.45.7 clang-1001.0.37.7)
Target: x86_64-apple-darwin18.2.0
ABI version: 0.6

Начнем с самых волнующих новостей:

Приложения на Swift больше не включают в себя DLL для стандартной библиотеки Swift и оверлеи Swift SDK в версиях для устройств на iOS 12.2, watchOS 5.2 и tvOS 12.2. В результате Swift-приложения уменьшаются в размере при развертывании для тестирования в TestFlight или при сжатии архива для дистрибутива.

ABI наконец-то станет стабильным, и это отличные новости. На мой взгляд, сегодня это одна из ключевых проблем Swift — не из-за побочных эффектов, а из-за невыполненных обещаний. Некоторые мои знакомые даже переписывают расширения для Apple Watch на Objective-C, чтобы уменьшить размер бинарного файла (примерно с 15 Мб до 1 Мб). Узнать больше о состоянии ABI можно по ссылкам: Информационная панель Swift ABI и Манифест стабильности Swift ABI.

Атрибут @dynamicCallable позволяет вызывать именованные типы так же, как функции, при помощи простого синтаксического сахара. Основное назначение — обеспечение совместимости с динамическими языками. (SE-0216)


Пример:

@dynamicCallable struct ToyCallable {
    func dynamicallyCall(withArguments: [Int]) {}
    func dynamicallyCall(withKeywordArguments: KeyValuePairs) {}
}

let x = ToyCallable()

x(1, 2, 3)
// Desugars to `x.dynamicallyCall(withArguments: [1, 2, 3])`

x(label: 1, 2)
// Desugars to `x.dynamicallyCall(withKeywordArguments: ["label": 1, "": 2])


Эта тема довольно обширна, и нововведение вызывает у меня смешанные чувства. Пол Хадсон сумел рассказать об этом с нейтральной позиции в статье «Что нового в Swift 5.0».

Режим совместимости со Swift 3 исключен. Поддерживаемые значения флага -swift-version — 4, 4.2 и 5.


Уходит эпоха: совместимости на уровне исходного кода со Swift 3 больше нет. Этот шаг был ожидаем, о нём было объявлено в Swift 5 Roadmap, и всё же. Очень советую освежить память, прочитав Swift 5.0 Release Process. Swift 5 уже рядом — будьте готовы.

В режиме Swift 5 switch-операторы над перечислениями, объявленными в Objective-C или приходящими из системных фреймворков, обязаны обрабатывать неизвестные значения — то есть те, которые могут быть добавлены в будущем или которые могут быть определены отдельно в файле реализации Objective-C.

Формально Objective-C позволяет включать в перечисление любое значение, если оно соответствует базовому типу. Эти неизвестные case«ы могут обрабатываться при помощи нового case«a @unknown default, который всё так же выдает предупреждение, если любые известные case«ы в switch окажутся упущены. Ими можно управлять и при помощи обычного default. Если вы задали перечисление в Objective-C и не хотите, чтобы клиентам пришлось обрабатывать неизвестные значения, воспользуйтесь макросом NS_CLOSED_ENUM вместо NS_ENUM. Компилятор Swift его распознает и не потребует задавать default в switch-операторе.

В режимах Swift 4 и 4.2 вы также можете пользоваться @unknown default. Если этого не сделать и пропустить в switch неизвестное значение, то выполнение программы прервётся — так же, как было в Swift 4.2 и Xcode 10.1 (SE-0192)


Это давняя проблема, по-прежнему доставляющая неудобства, особенно если вы предпочитаете не задавать в switch«ах default. Помню, сколько мучений мне доставила появившаяся в iOS 12 опция .provisional в типе UNAuthorizationOptions. Теперь, с введением case«а unknown, справляться с такими ситуациями стало куда проще.

Swift Package Manager


Теперь пакеты могут выбирать минимальную требуемую версию для платформы Apple (macOS, iOS) при использовании конфигурационного файла Package.swift из Swift 5. Сборка пакета невозможна, если хотя бы в одной из зависимостей минимальная версия платформы больше чем версия указанная для самого пакета. (SE-0236)


На мой взгляд, это самая главная новость, касающаяся SPM. По сути, эта фича может решить множество проблем, мешающих SPM стать полезным в iOS-мире. В своей прошлой статье я попробовал проанализировать текущее состояние SPM в контексте iOS-разработки. Кажется, теперь мне придётся пересмотреть сделанные выводы.

Но есть и проблемы:

  • в некоторых проектах наблюдается регрессия времени компиляции по сравнению с прошлыми релизами;
  • проекты для командной строки вылетают при запуске, выдавая ошибку dyld: Library not loaded
    (библиотека не загружена). Костыль: задайте пользовательскую настройку сборки SWIFT_FORCE_STATIC_LINK_STDLIB=YES.


В логе изменений перечислено множество исправленных неполадок и других сведений о Swift 5, которые могут быть важны для вашей сферы деятельности. Просмотрите их: возможно, вы решите использовать наследуемые назначенные инициализаторы с переменным количеством аргументов. Или вдруг вы столкнулись с проблемой взаимной блокировки из-за сложных рекурсивных определений типа с классами и дженериками. Или же псевдонимы дженериков в методе @objc доставляют вам неудобства.

Компилятор Clang


Появилось множество новых оповещений для компилятора Clang, и большинство из них касается фреймворков и модулей. Этот факт довольно интересен, поскольку он (предположительно) говорит об интеграции Swift Package Manager как инструмента управления зависимостями. Вот, по-моему, самые важные пункты:

  • новая диагностика идентифицирует заголовки фреймворка, которые используют включения с кавычками вместо включений в стиле фреймворка. По умолчанию оповещение выключено, но его можно включить, передав clang опцию -Wquoted-include-in-framework-header;
  • открытые заголовки фреймворка могут ошибочно выполнить #import или #include закрытых заголовков, что приводит к нарушению структуры, а иногда и к циклическому импорту модулей. О таких нарушениях сообщит новая диагностика. Она по умолчанию отключена в clang и управляется флагом -Wframework-include-private-from-public;
  • использование @import в заголовках фреймворка не дает использовать их без модулей. Новая диагностика выискивает @import в заголовках при передаче флага —fmodules. По умолчанию она отключена и управляется флагом -Watimport-in-framework-header;
  • раньше потеря ключевого слова framework при объявлении модуля для фреймворка не мешала компиляции, но молча приводила к неправильному результату. Новая диагностика -Wincomplete-framework-module-declaration и новый фикс предложат вам добавить нужный ключ. Это предупреждение запускается автоматически при передаче clang флага —fmodules.


Прежде всего: как их включить? Идите в меню Build Settings для вашего проекта, выберите пункт Apple Clang — Custom Compiler Flags и поставьте нужный флаг в пункте Other C Flags.

i-x2sfph0eovrw19ebbzs4cpyce.png

Я попробовал собрать старое приложение на Objective-C и обнаружил множество проблем с закрытыми заголовками в открытых заголовках фреймворка:

i-x2sfph0eovrw19ebbzs4cpyce.png

Также обнаружились проблемы со взятым в кавычки импортом внутри фреймворка:

utd2nrsy3zh2r_wrxi01mwsnoo8.png

Советую вам тоже провести подобную диагностику или хотя бы сделать заметку на будущее. Однажды эти проблемы могут доставить немало хлопот.

Система сборки


Есть и приятное обновление в системе сборки кода.

Поиск неявных зависимостей теперь работает и в разделе «другие флаги линкера» по связанным фреймворкам и библиотекам, помеченным -framework, -weak_framework, -reexport_framework, -lazy_framework, -weak-l, -reexport-l, -lazy-l или —l.


Это новшество также очень интригует. В сущности, оно означает, что теперь вы можете определять неявные зависимости с помощью опции .xcconfi или даже xcodebuild, избегая стадий Link / Embed в самом Xcode.

Отладка


Новшества отладчика:

Свойства UIStackView теперь показаны в инспекторе визуального отладчика. Визуальный отладчик получил более компактный трёхмерный вид.

vzqhirrk_uovikmjfsfn8vb_xt8.png

Теперь в случае возникновения ошибки из-за нехватки памяти Xcode может автоматически создавать график использования памяти. Вы можете включить создание графиков в окне диагностики в настройках выполнения схемы.

При приближении лимита использования памяти в iOS и watchOS Xcode отображает лимит в отчёте использования памяти.

j8rffqox3q4bamxpa_uurzu2np8.png

Видите красную черту? При её достижении Watchdog пришлёт уведомление applicationDidReceiveMemoryWarning. Я думал, что обновление окажется полезнее, а пока что это просто маленький плюсик.

Отладчик LLDB


Новые плюшки появились и в LLDB:

  • Теперь вы можете использовать $0, $1 и другие сокращения при вычислении выражений внутри замыканий.
  • Теперь в LLDB имеется новый псевдоним v для команды frame variable, служащий для вывода переменных в текущем стеке фрейма. Поскольку при этом не задействуется средство оценки выражений, v может быть намного быстрее, чем p или po, а потому предпочтительнее.


Я не обнаружил повышения производительности, однако в некоторых случаях v обеспечивает лучший вывод —, но при этом не является прямой заменой po и работает только с текущим стеком фрейма с определенными ограничениями. Примеры:

bo8bbigjksjdmb3bpeod_nmbvqs.png

Playgrounds


Мой любимый раздел! Начнём с багов:

Приложение Playgrounds может не запуститься


К несчастью, других новостей о Playgrounds в новой бете не поступало.

Симулятор


Пара слов о симуляторе:

Siri не работает в симуляторах watchOS и iOS.

Синхронизация Pasteboard между macOS и симулированными iOS-устройствами стала надёжнее.

Надеюсь, это действительно так.

Запрос доступа к микрофону на всех симулированных устройствах не будет повторяться.

Это хорошие новости, потому что из-за этой проблемы у многих возникали трудности с CI и агентами сборки. Теперь можно соорудить костыль, или, по крайней мере, дополнить свои руководства по работе с агентами сборки, включив пункт «Запустите симулятор один раз».

Тестирование


  • xccov поддерживает объединение нескольких отчётов о покрытии кода (и связанных с ними архивов) в один общий отчёт и архив. Получившийся отчёт может содержать неточности, поскольку между моментами создания отчётов могут произойти изменения в исходных файлах. Если изменений не произошло, отчёт будет верным.
  • Теперь xccov отражает разницу между разными отчётами Xcode о покрытии кода, что можно использовать для вычисления произошедших изменений в покрытии. Например, чтобы подсветить разницу между before.xccovreport и after.xccovreport, вызовите команду xccov следующим образом: xccov diff --json before.xccovreport after.xccovreport.
  • таргеты статических библиотек и фреймворков теперь появляются в отчёте о покрытии в виде записей верхнего уровня — графики показателей покрытия собираются по всем таргетам, включающим статическую библиотеку или фреймворк. При этом также решается проблема, когда исходные файлы статической библиотеки или таргета фреймворка вносятся в отчёт, даже в случае удаления самого таргета из покрытия кода в схеме.


Отличные новости для непрерывной интеграции, особенно демонстрация разницы. Расскажите своим релиз-инженерам или тем, кто занимается подобными вещами.

Впрочем, есть и несколько ограничений, касающихся распараллеливания тестов:

  • запись на клонах не работает при включенном распараллеливании;
  • профилирование проекта из Xcode ведет себя некорректно с включенным распараллеливанием тестов.

В остальном радует исправление некоторых багов:

Если тестирование провалится из-за падения прогонщика тестов при запуске, Xcode попробует сгенерировать подробное сообщение с описанием ошибки. Эту ошибку можно изучить в логе тестирования, а если вы пользуетесь xcodebuild, она появится и в stdout. Также она содержится в структурированных логах получившегося пакета.


У нас часто возникают подобные неполадки — и мы теряемся в догадках, что произошло. Иногда дело в неправильной компоновке, иногда в перегрузке системы. Теперь же надежность приложения повысится.

В собранных во время тестирования отчётах о вылете больше не теряются важные разделы вроде причины вылета или описания.


Здесь без комментариев, сплошной восторг.
Последняя новость, полезная многим разработчикам: теперь Xcode поддерживает сервис кэширования содержимого macOS. Это означает, что с вы можете создать сервер кэширования в локальной сети, который сэкономит время и деньги при загрузке новых и старых версий Xcode в локальной сети.

Проблемы


В этой бете я столкнулся с несколькими проблемами. В основном они касались инструментов от сторонних разработчиков. Например, carthage выдает ошибку «Не найдены доступные симуляторы для iOS».

Я проверил доступный симулятор, и, похоже, виноваты неполадки в бете. Кроме того, недоступна загрузка других рантаймов из Xcode — список доступных симуляторов пуст (a radar filled):

$ xcrun simctl list devices --json | grep -A16 12.1
    "com.apple.CoreSimulator.SimRuntime.iOS-12-1" : [
      {
        "availability" : "(unavailable, runtime profile not found)",
        "state" : "Shutdown",
        "isAvailable" : false,
        "name" : "iPhone 5s",
        "udid" : "DDD36346-A76F-42E8-80F4-6F11E1EE4BEB",
        "availabilityError" : "runtime profile not found"
      },
      {
        "availability" : "(unavailable, runtime profile not found)",
        "state" : "Shutdown",
        "isAvailable" : false,
        "name" : "iPhone 6",
        "udid" : "21794717-BC89-45E4-9F57-8CF9D14A87D1",
        "availabilityError" : "runtime profile not found"
      },
--


Но это ведь просто бета, а список изменений громаден. Запасёмся терпением. Carthage уже изучает ситуацию, все догадки изложены в этом баге.
Так-так. Похоже, что в Apple заняты полировкой технического долга и наложением патчей безопасности, так что никаких серьезных новостей, но есть две проблемы:

  • есть вероятность, что вы не сможете аутентифицироваться в Wallet после выбора карты;
  • есть вероятность, что вы не сможете приобрести предоплаченный тарифный план с помощью мобильной сети.


А еще Apple News теперь доступны в Канаде.

Не переключайтесь.


Единственное обновление здесь — возможная проблема с Safari 12.1 после апгрейда с Safari 10.1.2.

После апгрейда до Safari 12.1 с версии Safari 10.1.2 веб-страницы могут отображаться некорректно. (47335741)


Обходной прием: введите в терминале команду defaults delete com.apple.Safari. Обратите внимание, что при этом вы потеряете все прошлые настройки Safari.
Статья вышла куда объёмнее, чем я рассчитывал. Все мои выводы вы найдёте выше. А если вкратце — Swift 5 пришёл!

Спасибо за внимание.

© Habrahabr.ru