shopify_toolkit 0.1.0.pre → 0.1.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 +4 -4
- data/README.md +84 -2
- data/exe/shopify-toolkit +59 -0
- data/lib/shopify_toolkit/admin_client.rb +39 -0
- data/lib/shopify_toolkit/metafield_statements.rb +143 -0
- data/lib/shopify_toolkit/migration/logging.rb +29 -0
- data/lib/shopify_toolkit/version.rb +1 -1
- data/lib/shopify_toolkit.rb +8 -2
- metadata +65 -4
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: e21a90f2ee3abbe3ca01f52a8b3f559437d357e41e649d2351a5740c83dfcb1c
         | 
| 4 | 
            +
              data.tar.gz: 5e731cba7ace9bf63e2e68203efe6a12db03642a2fbe88d3db34ed91cc903450
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 87dd02b8491e762cf1e3ffa8e2cada8b8c243d921b35e69f7978dad66d333fe767b463b61e9920600bfbd312c438734342adc6306d656efe07efa63bab918b83
         | 
| 7 | 
            +
              data.tar.gz: e1a5daad1b1dd2f8239f8e2c45b5b988c8864a9536533656fc76a66cd7c420df9be66bc441f58e5a1786019bac4d284e515367d6aa3d1b3ce672f0354023fdbc
         | 
    
        data/README.md
    CHANGED
    
    | @@ -4,8 +4,10 @@ A toolkit for working with Custom Shopify Apps built on Rails. | |
| 4 4 |  | 
| 5 5 | 
             
            ## Features/Roadmap
         | 
| 6 6 |  | 
| 7 | 
            +
            - [x] Shopify/Matrixify CSV tools
         | 
| 7 8 | 
             
            - [ ] Metafield/Metaobject migrations (just like ActiveRecord migrations, but for Shopify!)
         | 
| 8 | 
            -
            - [ | 
| 9 | 
            +
              - [x] Metafield Definitions management API
         | 
| 10 | 
            +
              - [ ] Metaobject Definitions management API
         | 
| 9 11 | 
             
            - [ ] GraphQL Admin API code generation (syntax checking, etc)
         | 
| 10 12 | 
             
            - [ ] GraphQL Admin API client with built-in rate limiting
         | 
| 11 13 | 
             
            - [ ] GraphQL Admin API client with built-in caching
         | 
| @@ -22,7 +24,87 @@ bundle add shopify_toolkit | |
| 22 24 |  | 
| 23 25 | 
             
            ## Usage
         | 
| 24 26 |  | 
| 25 | 
            -
             | 
| 27 | 
            +
            ### Migrating Metafields definitions using ActiveRecord Migrations
         | 
| 28 | 
            +
             | 
| 29 | 
            +
            Within a Rails application created with ShopifyApp, generate a new migration file:
         | 
| 30 | 
            +
             | 
| 31 | 
            +
            ```bash
         | 
| 32 | 
            +
            rails generate migration AddMetafieldDefinitions
         | 
| 33 | 
            +
            ```
         | 
| 34 | 
            +
             | 
| 35 | 
            +
            Include the `ShopifyToolkit::MetafieldStatements` module in your migration file
         | 
| 36 | 
            +
            in order to use the metafield statements:
         | 
| 37 | 
            +
             | 
| 38 | 
            +
            ```ruby
         | 
| 39 | 
            +
            class AddMetafieldDefinitions < ActiveRecord::Migration[7.0]
         | 
| 40 | 
            +
              include ShopifyToolkit::MetafieldStatements
         | 
| 41 | 
            +
             | 
| 42 | 
            +
              def up
         | 
| 43 | 
            +
                Shop.first!.with_shopify_session do
         | 
| 44 | 
            +
                  create_metafield :products, :my_metafield, :single_line_text_field, name: "My Metafield"
         | 
| 45 | 
            +
                end
         | 
| 46 | 
            +
              end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
              def down
         | 
| 49 | 
            +
                Shop.first!.with_shopify_session do
         | 
| 50 | 
            +
                  remove_metafield :products, :my_metafield
         | 
| 51 | 
            +
                end
         | 
| 52 | 
            +
              end
         | 
| 53 | 
            +
            end
         | 
| 54 | 
            +
            ```
         | 
| 55 | 
            +
            Then run the migration:
         | 
| 56 | 
            +
             | 
| 57 | 
            +
            ```bash
         | 
| 58 | 
            +
            rails db:migrate
         | 
| 59 | 
            +
            ```
         | 
| 60 | 
            +
             | 
| 61 | 
            +
            ### Creating a Metafield Schema Definition
         | 
| 62 | 
            +
             | 
| 63 | 
            +
            You can also create a metafield schema definition file to define your metafields in a more structured way. This is useful for keeping track of your metafields and their definitions.
         | 
| 64 | 
            +
             | 
| 65 | 
            +
            ```rb
         | 
| 66 | 
            +
            # config/shopify/schema.rb
         | 
| 67 | 
            +
             | 
| 68 | 
            +
            ShopifyToolkit::MetafieldSchema.define do
         | 
| 69 | 
            +
              # Define your metafield schema here
         | 
| 70 | 
            +
              # For example:
         | 
| 71 | 
            +
              create_metafield :products, :my_metafield, :single_line_text_field, name: "My Metafield"
         | 
| 72 | 
            +
            end
         | 
| 73 | 
            +
            ```
         | 
| 74 | 
            +
             | 
| 75 | 
            +
            ### Analyzing a Matrixify CSV Result files
         | 
| 76 | 
            +
             | 
| 77 | 
            +
            Matrixify is a popular Shopify app that allows you to import/export data from Shopify using CSV files. The CSV files that Matrixify generates are very verbose and can be difficult to work with. This tool allows you to analyze the CSV files and extract the data you need.
         | 
| 78 | 
            +
             | 
| 79 | 
            +
            The tool will import the file into a local SQLite database
         | 
| 80 | 
            +
            and open a console for you to run queries against the data
         | 
| 81 | 
            +
            using ActiveRecord.
         | 
| 82 | 
            +
             | 
| 83 | 
            +
            ```shell
         | 
| 84 | 
            +
            shopify-csv analyze products-result.csv --force-import
         | 
| 85 | 
            +
            ==> Importing products-result.csv into /var/folders/hn/z7b7s1kj3js4k7_qk3lj27kr0000gn/T/shopify-toolkit-analyze-products_result_csv.sqlite3
         | 
| 86 | 
            +
            -- create_table(:results, {:force=>true})
         | 
| 87 | 
            +
               -> 0.0181s
         | 
| 88 | 
            +
            -- add_index(:results, :import_result)
         | 
| 89 | 
            +
               -> 0.0028s
         | 
| 90 | 
            +
            ........................
         | 
| 91 | 
            +
            ==> Starting console for products-result.csv
         | 
| 92 | 
            +
            >> comments
         | 
| 93 | 
            +
            =>
         | 
| 94 | 
            +
            ["UPDATE: Found by Handle | Assuming MERGE for Variant Command | Assuming MERGE for Tags Command | Variants updated by SKU: 1",
         | 
| 95 | 
            +
             "UPDATE: Found by Handle | Assuming MERGE for Variant Command | Assuming MERGE for Tags Command | Variants updated by SKU: 1 | Warning: The following media were not uploaded to Shopify: [https://mymedia.com/image.jpg: Error downloading from Web: 302 Moved Temporarily]",
         | 
| 96 | 
            +
             ...]
         | 
| 97 | 
            +
            >> first
         | 
| 98 | 
            +
            #<ShopifyCSV::Result:0x0000000300bf1668
         | 
| 99 | 
            +
             id: 1,
         | 
| 100 | 
            +
             data: nil,
         | 
| 101 | 
            +
             handle: "my-product",
         | 
| 102 | 
            +
             title: "My Product",
         | 
| 103 | 
            +
             import_result: "OK",
         | 
| 104 | 
            +
             import_comment: "UPDATE: Found by Handle | Assuming MERGE for Varia...">
         | 
| 105 | 
            +
            >> count
         | 
| 106 | 
            +
            => 116103
         | 
| 107 | 
            +
            ```
         | 
| 26 108 |  | 
| 27 109 | 
             
            ## Development
         | 
| 28 110 |  | 
    
        data/exe/shopify-toolkit
    ADDED
    
    | @@ -0,0 +1,59 @@ | |
| 1 | 
            +
            #!/usr/bin/env ruby
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require 'csv'
         | 
| 4 | 
            +
            require 'active_record'
         | 
| 5 | 
            +
            require 'thor'
         | 
| 6 | 
            +
            require 'tmpdir'
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            class ShopifyToolkit::CommandLine < Thor
         | 
| 9 | 
            +
              RESERVED_COLUMN_NAMES = %w[select type id]
         | 
| 10 | 
            +
             | 
| 11 | 
            +
              class Result < ActiveRecord::Base
         | 
| 12 | 
            +
                def self.comments
         | 
| 13 | 
            +
                  distinct.pluck(:import_comment)
         | 
| 14 | 
            +
                end
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                def self.with_comment(text)
         | 
| 17 | 
            +
                  where("import_comment LIKE ?", "%#{text}%")
         | 
| 18 | 
            +
                end
         | 
| 19 | 
            +
              end
         | 
| 20 | 
            +
             | 
| 21 | 
            +
              desc "analyze CSV_PATH", "Analyze results file at path CSV_PATH"
         | 
| 22 | 
            +
              method_option :force_import, type: :boolean, default: false
         | 
| 23 | 
            +
              method_option :tmp_dir, type: :string, default: Dir.tmpdir
         | 
| 24 | 
            +
              def analyze(csv_path)
         | 
| 25 | 
            +
                csv_path = File.expand_path(csv_path)
         | 
| 26 | 
            +
                underscore = ->(string) { string.downcase.gsub(/[^a-z0-9]+/, "_").gsub(/(^_+|_+$)/, "") }
         | 
| 27 | 
            +
                csv = CSV.open(csv_path, liberal_parsing:true )
         | 
| 28 | 
            +
                header_to_column = -> { RESERVED_COLUMN_NAMES.include?(_1.to_s) ? "#{_1}_1" : _1 }
         | 
| 29 | 
            +
                headers = csv.shift.map(&underscore).map(&header_to_column)
         | 
| 30 | 
            +
                basename = File.basename csv_path
         | 
| 31 | 
            +
                database = "#{options[:tmp_dir]}/shopify-toolkit-analyze-#{underscore[basename]}.sqlite3"
         | 
| 32 | 
            +
                should_import = options[:force_import] || !File.exist?(database)
         | 
| 33 | 
            +
                to_record = ->(row) { headers.zip(row.each{ |c| c.delete!("\u0000") if String === c }).to_h.transform_keys(&header_to_column) }
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                File.delete(database) if should_import && File.exist?(database)
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                ActiveRecord::Base.establish_connection(adapter: 'sqlite3', database:)
         | 
| 38 | 
            +
             | 
| 39 | 
            +
                if should_import
         | 
| 40 | 
            +
                  puts "==> Importing #{csv_path} into #{database}"
         | 
| 41 | 
            +
                  ActiveRecord::Schema.define do
         | 
| 42 | 
            +
                    create_table :results, force: true do |t|
         | 
| 43 | 
            +
                      t.json :data
         | 
| 44 | 
            +
                      headers.each { |header| t.string header }
         | 
| 45 | 
            +
                    end
         | 
| 46 | 
            +
                    add_index :results, :import_result if headers.include?('import_result')
         | 
| 47 | 
            +
                  end
         | 
| 48 | 
            +
                  csv.each_slice(5000) { |rows| print "."; Result.insert_all(rows.map(&to_record)) }
         | 
| 49 | 
            +
                  puts
         | 
| 50 | 
            +
                end
         | 
| 51 | 
            +
             | 
| 52 | 
            +
                puts "==> Starting console for #{basename}"
         | 
| 53 | 
            +
                require "irb"
         | 
| 54 | 
            +
                IRB.conf[:IRB_NAME] = basename
         | 
| 55 | 
            +
                Result.class_eval { binding.irb(show_code: false) }
         | 
| 56 | 
            +
              end
         | 
| 57 | 
            +
            end
         | 
| 58 | 
            +
             | 
| 59 | 
            +
            ShopifyToolkit::CommandLine.start(ARGV)
         | 
| @@ -0,0 +1,39 @@ | |
| 1 | 
            +
            module ShopifyToolkit::AdminClient
         | 
| 2 | 
            +
              API_VERSION = "2024-10"
         | 
| 3 | 
            +
             | 
| 4 | 
            +
              def api_version
         | 
| 5 | 
            +
                API_VERSION
         | 
| 6 | 
            +
              end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
              def shopify_admin_client
         | 
| 9 | 
            +
                @shopify_admin_client ||=
         | 
| 10 | 
            +
                  ShopifyAPI::Clients::Graphql::Admin.new(session: ShopifyAPI::Context.active_session, api_version:)
         | 
| 11 | 
            +
              end
         | 
| 12 | 
            +
             | 
| 13 | 
            +
              def handle_shopify_admin_client_errors(response, *user_error_paths)
         | 
| 14 | 
            +
                if response.code != 200
         | 
| 15 | 
            +
                  logger.error "Error querying Shopify Admin API: #{response.inspect}"
         | 
| 16 | 
            +
                  raise "Error querying Shopify Admin API: #{response.inspect}"
         | 
| 17 | 
            +
                end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                response
         | 
| 20 | 
            +
                  .body
         | 
| 21 | 
            +
                  .dig("errors")
         | 
| 22 | 
            +
                  .to_a
         | 
| 23 | 
            +
                  .each do |error|
         | 
| 24 | 
            +
                    logger.error "Error querying Shopify Admin API: #{error.inspect}"
         | 
| 25 | 
            +
                    raise "Error querying Shopify Admin API: #{error.inspect}"
         | 
| 26 | 
            +
                  end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                user_error_paths.each do |path|
         | 
| 29 | 
            +
                  response
         | 
| 30 | 
            +
                    .body
         | 
| 31 | 
            +
                    .dig(*path.split("."))
         | 
| 32 | 
            +
                    .to_a
         | 
| 33 | 
            +
                    .each do |error|
         | 
| 34 | 
            +
                      logger.error "Error querying Shopify Admin API: #{error.inspect} (#{path})"
         | 
| 35 | 
            +
                      raise "Error querying Shopify Admin API: #{error.inspect} (#{path})"
         | 
| 36 | 
            +
                    end
         | 
| 37 | 
            +
                end
         | 
| 38 | 
            +
              end
         | 
| 39 | 
            +
            end
         | 
| @@ -0,0 +1,143 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "shopify_toolkit/migration/logging"
         | 
| 4 | 
            +
            require "shopify_toolkit/admin_client"
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            module ShopifyToolkit::MetafieldStatements
         | 
| 7 | 
            +
              extend ActiveSupport::Concern
         | 
| 8 | 
            +
              include ShopifyToolkit::Migration::Logging
         | 
| 9 | 
            +
              include ShopifyToolkit::AdminClient
         | 
| 10 | 
            +
             | 
| 11 | 
            +
              def self.log_time(method_name)
         | 
| 12 | 
            +
                current_method = instance_method(method_name)
         | 
| 13 | 
            +
                define_method(method_name) do |*args, **kwargs, &block|
         | 
| 14 | 
            +
                  say_with_time("#{method_name}(#{args.map(&:inspect).join(', ')})") { current_method.bind(self).call(*args, **kwargs, &block) }
         | 
| 15 | 
            +
                end
         | 
| 16 | 
            +
              end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
              # create_metafield :products, :my_metafield, :single_line_text_field, name: "Prova"
         | 
| 19 | 
            +
              # @param namespace: if nil the metafield will be app-specific (default: :custom)
         | 
| 20 | 
            +
              log_time \
         | 
| 21 | 
            +
              def create_metafield(owner_type, key, type, namespace: :custom, name:, **options)
         | 
| 22 | 
            +
                ownerType = owner_type.to_s.singularize.upcase # Eg. "PRODUCT"
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                # https://shopify.dev/docs/api/admin-graphql/2024-10/mutations/metafieldDefinitionCreate
         | 
| 25 | 
            +
                query =
         | 
| 26 | 
            +
                  "# GraphQL
         | 
| 27 | 
            +
                    mutation CreateMetafieldDefinition($definition: MetafieldDefinitionInput!) {
         | 
| 28 | 
            +
                      metafieldDefinitionCreate(definition: $definition) {
         | 
| 29 | 
            +
                        createdDefinition {
         | 
| 30 | 
            +
                          id
         | 
| 31 | 
            +
                          key
         | 
| 32 | 
            +
                        }
         | 
| 33 | 
            +
                        userErrors {
         | 
| 34 | 
            +
                          field
         | 
| 35 | 
            +
                          message
         | 
| 36 | 
            +
                          code
         | 
| 37 | 
            +
                        }
         | 
| 38 | 
            +
                      }
         | 
| 39 | 
            +
                    }
         | 
| 40 | 
            +
                  "
         | 
| 41 | 
            +
                variables = { definition: { ownerType:, key:, type:, name:, namespace:, **options } }
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                shopify_admin_client
         | 
| 44 | 
            +
                  .query(query:, variables:)
         | 
| 45 | 
            +
                  .tap { handle_shopify_admin_client_errors(_1, "metafieldDefinitionCreate.userErrors") }
         | 
| 46 | 
            +
              end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
              def get_metafield_gid(owner_type, key, namespace: :custom)
         | 
| 49 | 
            +
                ownerType = owner_type.to_s.singularize.upcase # Eg. "PRODUCT"
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                result =
         | 
| 52 | 
            +
                  shopify_admin_client
         | 
| 53 | 
            +
                    .query(
         | 
| 54 | 
            +
                      query:
         | 
| 55 | 
            +
                        "# GraphQL
         | 
| 56 | 
            +
                          query FindMetafieldDefinition($ownerType: MetafieldOwnerType!, $key: String!) {
         | 
| 57 | 
            +
                            metafieldDefinitions(first: 1, ownerType: $ownerType, key: $key) {
         | 
| 58 | 
            +
                              nodes { id }
         | 
| 59 | 
            +
                            }
         | 
| 60 | 
            +
                          }",
         | 
| 61 | 
            +
                      variables: {
         | 
| 62 | 
            +
                        ownerType:,
         | 
| 63 | 
            +
                        key:,
         | 
| 64 | 
            +
                        namespace:,
         | 
| 65 | 
            +
                      },
         | 
| 66 | 
            +
                    )
         | 
| 67 | 
            +
                    .tap { handle_shopify_admin_client_errors(_1) }
         | 
| 68 | 
            +
                    .body
         | 
| 69 | 
            +
             | 
| 70 | 
            +
                result.dig("data", "metafieldDefinitions", "nodes", 0, "id") or
         | 
| 71 | 
            +
                  raise "Metafield not found for #{owner_type}##{namespace}:#{key}"
         | 
| 72 | 
            +
              end
         | 
| 73 | 
            +
             | 
| 74 | 
            +
              log_time \
         | 
| 75 | 
            +
              def remove_metafield(owner_type, key, namespace: :custom, delete_associated_metafields: false, **options)
         | 
| 76 | 
            +
                if namespace == nil && delete_associated_metafields == false
         | 
| 77 | 
            +
                  raise ArgumentError,
         | 
| 78 | 
            +
                        "For reserved namespaces, you must delete all associated metafields (delete_associated_metafields: true)"
         | 
| 79 | 
            +
                end
         | 
| 80 | 
            +
             | 
| 81 | 
            +
                shopify_admin_client
         | 
| 82 | 
            +
                  .query(
         | 
| 83 | 
            +
                    # Documentation: https://shopify.dev/docs/api/admin-graphql/2024-10/mutations/metafieldDefinitionDelete
         | 
| 84 | 
            +
                    query:
         | 
| 85 | 
            +
                      "# GraphQL
         | 
| 86 | 
            +
                        mutation DeleteMetafieldDefinition($id: ID!, $deleteAllAssociatedMetafields: Boolean!) {
         | 
| 87 | 
            +
                          metafieldDefinitionDelete(id: $id, deleteAllAssociatedMetafields: $deleteAllAssociatedMetafields) {
         | 
| 88 | 
            +
                            deletedDefinitionId
         | 
| 89 | 
            +
                            userErrors {
         | 
| 90 | 
            +
                              field
         | 
| 91 | 
            +
                              message
         | 
| 92 | 
            +
                              code
         | 
| 93 | 
            +
                            }
         | 
| 94 | 
            +
                          }
         | 
| 95 | 
            +
                        }",
         | 
| 96 | 
            +
                    variables: {
         | 
| 97 | 
            +
                      id: get_metafield_gid(owner_type, key, namespace: namespace),
         | 
| 98 | 
            +
                      deleteAllAssociatedMetafields: delete_associated_metafields,
         | 
| 99 | 
            +
                    },
         | 
| 100 | 
            +
                  )
         | 
| 101 | 
            +
                  .tap { handle_shopify_admin_client_errors(_1, "metafieldDefinitionDelete.userErrors") }
         | 
| 102 | 
            +
              end
         | 
| 103 | 
            +
             | 
| 104 | 
            +
              def update_metafield(owner_type, key, namespace: :custom, **options)
         | 
| 105 | 
            +
              log_time \
         | 
| 106 | 
            +
                shopify_admin_client
         | 
| 107 | 
            +
                  .query(
         | 
| 108 | 
            +
                    # Documentation: https://shopify.dev/docs/api/admin-graphql/2024-10/mutations/metafieldDefinitionUpdate
         | 
| 109 | 
            +
                    query:
         | 
| 110 | 
            +
                      "# GraphQL
         | 
| 111 | 
            +
                        mutation UpdateMetafieldDefinition($definition: MetafieldDefinitionUpdateInput!) {
         | 
| 112 | 
            +
                          metafieldDefinitionUpdate(definition: $definition) {
         | 
| 113 | 
            +
                            updatedDefinition {
         | 
| 114 | 
            +
                              id
         | 
| 115 | 
            +
                              name
         | 
| 116 | 
            +
                            }
         | 
| 117 | 
            +
                            userErrors {
         | 
| 118 | 
            +
                              field
         | 
| 119 | 
            +
                              message
         | 
| 120 | 
            +
                              code
         | 
| 121 | 
            +
                            }
         | 
| 122 | 
            +
                          }
         | 
| 123 | 
            +
                        }",
         | 
| 124 | 
            +
                    variables: {
         | 
| 125 | 
            +
                      definition: {
         | 
| 126 | 
            +
                        ownerType: owner_type.to_s.singularize.upcase,
         | 
| 127 | 
            +
                        key:,
         | 
| 128 | 
            +
                        namespace:,
         | 
| 129 | 
            +
                        **options,
         | 
| 130 | 
            +
                      },
         | 
| 131 | 
            +
                    },
         | 
| 132 | 
            +
                  )
         | 
| 133 | 
            +
                  .tap { handle_shopify_admin_client_errors(_1, "metafieldDefinitionUpdate.userErrors") }
         | 
| 134 | 
            +
              end
         | 
| 135 | 
            +
             | 
| 136 | 
            +
              def self.define(&block)
         | 
| 137 | 
            +
                context = Object.new
         | 
| 138 | 
            +
                context.extend(self)
         | 
| 139 | 
            +
             | 
| 140 | 
            +
                context.instance_eval(&block) if block_given?(&block)
         | 
| 141 | 
            +
                context
         | 
| 142 | 
            +
              end
         | 
| 143 | 
            +
            end
         | 
| @@ -0,0 +1,29 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ShopifyToolkit::Migration::Logging
         | 
| 4 | 
            +
              def write(text = "")
         | 
| 5 | 
            +
                puts(text)
         | 
| 6 | 
            +
              end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
              def announce(message)
         | 
| 9 | 
            +
                text = "#{version} #{name}: #{message}"
         | 
| 10 | 
            +
                length = [0, 75 - text.length].max
         | 
| 11 | 
            +
                write "== %s %s" % [text, "=" * length]
         | 
| 12 | 
            +
              end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
              # Takes a message argument and outputs it as is.
         | 
| 15 | 
            +
              # A second boolean argument can be passed to specify whether to indent or not.
         | 
| 16 | 
            +
              def say(message, subitem = false)
         | 
| 17 | 
            +
                write "#{subitem ? "   ->" : "--"} #{message}"
         | 
| 18 | 
            +
              end
         | 
| 19 | 
            +
             | 
| 20 | 
            +
              # Outputs text along with how long it took to run its block.
         | 
| 21 | 
            +
              # If the block returns an integer it assumes it is the number of rows affected.
         | 
| 22 | 
            +
              def say_with_time(message)
         | 
| 23 | 
            +
                say(message)
         | 
| 24 | 
            +
                result = nil
         | 
| 25 | 
            +
                time_elapsed = ActiveSupport::Benchmark.realtime { result = yield }
         | 
| 26 | 
            +
                say "%.4fs" % time_elapsed, :subitem
         | 
| 27 | 
            +
                result
         | 
| 28 | 
            +
              end
         | 
| 29 | 
            +
            end
         | 
    
        data/lib/shopify_toolkit.rb
    CHANGED
    
    | @@ -1,8 +1,14 @@ | |
| 1 1 | 
             
            # frozen_string_literal: true
         | 
| 2 2 |  | 
| 3 3 | 
             
            require_relative "shopify_toolkit/version"
         | 
| 4 | 
            +
            require "zeitwerk"
         | 
| 4 5 |  | 
| 5 6 | 
             
            module ShopifyToolkit
         | 
| 6 | 
            -
             | 
| 7 | 
            -
               | 
| 7 | 
            +
             | 
| 8 | 
            +
              def self.loader
         | 
| 9 | 
            +
                @loader ||= Zeitwerk::Loader.for_gem
         | 
| 10 | 
            +
              end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
              loader.setup
         | 
| 13 | 
            +
              # loader.eager_load # optionally
         | 
| 8 14 | 
             
            end
         | 
    
        metadata
    CHANGED
    
    | @@ -1,18 +1,75 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: shopify_toolkit
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 0.1.0 | 
| 4 | 
            +
              version: 0.1.0
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Elia Schito
         | 
| 8 8 | 
             
            - Nebulab Team
         | 
| 9 9 | 
             
            bindir: exe
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date: 2025- | 
| 12 | 
            -
            dependencies: | 
| 11 | 
            +
            date: 2025-04-02 00:00:00.000000000 Z
         | 
| 12 | 
            +
            dependencies:
         | 
| 13 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 14 | 
            +
              name: thor
         | 
| 15 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 16 | 
            +
                requirements:
         | 
| 17 | 
            +
                - - ">="
         | 
| 18 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 19 | 
            +
                    version: '1.3'
         | 
| 20 | 
            +
              type: :runtime
         | 
| 21 | 
            +
              prerelease: false
         | 
| 22 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 23 | 
            +
                requirements:
         | 
| 24 | 
            +
                - - ">="
         | 
| 25 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 26 | 
            +
                    version: '1.3'
         | 
| 27 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 28 | 
            +
              name: zeitwerk
         | 
| 29 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 30 | 
            +
                requirements:
         | 
| 31 | 
            +
                - - ">="
         | 
| 32 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 33 | 
            +
                    version: '2.7'
         | 
| 34 | 
            +
              type: :runtime
         | 
| 35 | 
            +
              prerelease: false
         | 
| 36 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 37 | 
            +
                requirements:
         | 
| 38 | 
            +
                - - ">="
         | 
| 39 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 40 | 
            +
                    version: '2.7'
         | 
| 41 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 42 | 
            +
              name: activesupport
         | 
| 43 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 44 | 
            +
                requirements:
         | 
| 45 | 
            +
                - - ">="
         | 
| 46 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 47 | 
            +
                    version: '7.0'
         | 
| 48 | 
            +
              type: :runtime
         | 
| 49 | 
            +
              prerelease: false
         | 
| 50 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 51 | 
            +
                requirements:
         | 
| 52 | 
            +
                - - ">="
         | 
| 53 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 54 | 
            +
                    version: '7.0'
         | 
| 55 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 56 | 
            +
              name: shopify_api
         | 
| 57 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 58 | 
            +
                requirements:
         | 
| 59 | 
            +
                - - ">="
         | 
| 60 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 61 | 
            +
                    version: '14.8'
         | 
| 62 | 
            +
              type: :runtime
         | 
| 63 | 
            +
              prerelease: false
         | 
| 64 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 65 | 
            +
                requirements:
         | 
| 66 | 
            +
                - - ">="
         | 
| 67 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 68 | 
            +
                    version: '14.8'
         | 
| 13 69 | 
             
            email:
         | 
| 14 70 | 
             
            - elia@schito.me
         | 
| 15 | 
            -
            executables: | 
| 71 | 
            +
            executables:
         | 
| 72 | 
            +
            - shopify-toolkit
         | 
| 16 73 | 
             
            extensions: []
         | 
| 17 74 | 
             
            extra_rdoc_files: []
         | 
| 18 75 | 
             
            files:
         | 
| @@ -20,7 +77,11 @@ files: | |
| 20 77 | 
             
            - LICENSE.txt
         | 
| 21 78 | 
             
            - README.md
         | 
| 22 79 | 
             
            - Rakefile
         | 
| 80 | 
            +
            - exe/shopify-toolkit
         | 
| 23 81 | 
             
            - lib/shopify_toolkit.rb
         | 
| 82 | 
            +
            - lib/shopify_toolkit/admin_client.rb
         | 
| 83 | 
            +
            - lib/shopify_toolkit/metafield_statements.rb
         | 
| 84 | 
            +
            - lib/shopify_toolkit/migration/logging.rb
         | 
| 24 85 | 
             
            - lib/shopify_toolkit/version.rb
         | 
| 25 86 | 
             
            - sig/shopify_toolkit.rbs
         | 
| 26 87 | 
             
            homepage: https://github.com/nebulab/shopify_toolkit?tab=readme-ov-file#readme
         |