Various embodiments of the present disclosure are generally directed to an apparatus and method for adding storage capacity to an object storage system, such as used in a cloud computing environment.
In some embodiments, a first set of data storage devices store data objects in accordance with a first map structure. A management module detects a second set of data storage devices added to the first set and, in response thereto, generates a second map structure and migrates a portion of the data objects from the first set to the second set based on the second map structure to balance the first and second sets.
In further embodiments, a storage controller has a processor and a memory. A plurality of storage devices is connected to the storage controller to form a storage node. The storage devices within the node are arranged into N subgroups, so that data objects are stored in the N subgroups based on a first map structure stored in the storage controller memory. A management module detects an additional plurality of storage devices that are connected to the storage controller to provide N+1 subgroups. In response, the management module migrates a portion of the data objects in each of the N subgroups to the additional plurality of storage devices, generates a second map structure to describe the data objects stored in the N+1 subgroups, and stores the second map structure in the storage controller memory.
In further embodiments, a computer implemented method includes storing data objects in a first set of data storage devices of an object storage system in accordance with a first map structure stored in a memory; connecting a second set of data storage devices to the first set; and detecting the connection of the second set to the first set, and in response thereto, generating a second map structure and migrating a portion of the data objects from the first set to the second set based on the second map structure to balance the first and second sets.
The present disclosure generally relates to the automated addition of new storage capacity to an object storage system, such as in a cloud computing environment.
Cloud computing generally refers to a network-based distributed data processing environment. Network services such as computational resources, software and/or data are made available to remote users via a wide area network, such as but not limited to the Internet. In other embodiments, the resources can be made available to local user via a localized network. A cloud computing network can be a public “available-by-subscription” service accessible by substantially any user for a fee, or a private “in-house” service operated by or for the use of one or more dedicated users.
A cloud computing network can be generally arranged as an object storage system whereby data objects (e.g., files) from users (“account holders” or simply “accounts”) are replicated and can be stored in geographically distributed storage locations or other beneficial organization structures within the system. The network is often accessed through web-based tools such as web browsers, and provides services to a user as if such services were installed locally on the user's local computer.
Attempts have been made to configure object storage systems to be massively scalable so that new storage nodes, servers, software modules, etc. can be added to the system to expand overall capabilities. An object storage system can carry out significant amounts of background overhead processing to store, replicate, migrate and rebalance the data objects stored within the system in an effort to ensure the data objects are available to the users at all times. The addition of new storage capacity to an existing object storage system can present a number of challenges with regard to ensuring system integrity, reliability and availability levels are maintained throughout the expansion process.
Accordingly, various embodiments of the present disclosure are generally directed to the addition of storage capacity to an existing storage cluster or other collective unit of an object storage system. In some embodiments, the storage cluster may be a SwiftStack cluster. As explained below, a server or proxy server is adapted to communicate with users of an object storage system over a computer network. A plurality of data storage devices store and retrieve data objects from the users. The data storage devices may be arranged into a plurality of storage nodes each having an associated storage controller. Map structures are used to associate storage entities such as the data objects with physical locations within the data storage devices.
The expansion of the existing data storage capacity of the system can be carried out to expand the storage capacity of an existing node and/or to add one or more new storage nodes to the system. A storage management module is configured to detect new storage, generate new mapping to describe the new storage, to migrate data objects within the system and to deploy the new mapping. In some embodiments, the data migration is carried out prior to the deployment of the new mapping. In other embodiments, the new mapping is deployed and the data objects are migrated to conform the system to the new mapping.
In some embodiments, a system administrator provides administrative input data to the system in conjunction with the addition of the new storage capacity to the system. The storage management module uses the existing mapping and the administrative input data to generate the new mapping and to commence transfer of at least some data objects to the new storage capacity.
In further embodiments, existing system I/O traffic with users of the system is monitored and the data migration operations are scheduled to fit within the available bandwidth of the system. In still further embodiments, secondary data migration paths are utilized to transfer data to the new storage locations.
The map structures may be referred to as rings, and the rings may be arranged as an account ring, a container ring and an object ring. The account ring provides lists of containers, or groups of data objects owned by a particular user (“account”). The container ring provides lists of data objects in each container, and the object ring provides lists of data objects mapped to their particular storage locations. Other forms of map structures can be used.
By automatically detecting and utilizing new storage, the system efficiently and effectively rebalances the system. The data objects can be quickly and efficiently migrated in the background without substantively affecting user data access operations or overhead processing within the system. In some embodiments, the detection and utilization will include some limited user input or answering of minor questions; while other embodiments will not include such an interaction.
These and various other features of various embodiments disclosed herein can be understood beginning with a review of
The system 100 is accessed by one or more user devices 102, which may take the form of a network accessible device such as a desktop computer, a terminal, a laptop, a tablet, a smartphone, a game console or other device with network connectivity capabilities. In some cases, each user device 102 accesses the system 100 via a web-based application on the user device that communicates with the system 100 over a network 104. The network 104 may take the form of the Internet or some other computer-based network.
The system 100 includes various elements that may be distributed over a large area. These elements include one or more management servers 106 which process communications with the user devices 102 and perform other system functions. A plurality of storage controllers 108 control local groups of storage devices 110 used to store data objects from the user devices 102, and to return the data data objects as requested. Each grouping of storage devices 110 and associated controller 108 is characterized as a storage node 112.
While only three storage nodes 112 are illustrated in
Generally, data presented to the system 100 by the users of the system are organized as data objects, each constituting a cohesive associated data set (e.g., a file) having an object identifier (e.g., a “name” or “category”). Examples include databases, word processing and other application files, graphics, A/V works, web pages, games, executable programs, etc. Substantially any type of data object can be stored depending on the parametric configuration of the system.
Each data object presented to the system 100 will be subjected to a system replication policy so that multiple copies of the data object are stored in different zones. It is contemplated albeit not required that the system nominally generates and stores three replicas of each data object. This enhances data reliability, but generally increases background overhead processing to maintain the system in an updated state.
An example hardware architecture for portions of the system 100 is represented in
In some cases, the functionality of the storage controller 108 can be carried out using the local computer 116. In other cases, the storage controller functionality carried out by processing capabilities of one or more of the storage enclosures 120, and the computer 116 can be eliminated or used for other purposes such as local administrative personnel access. In one embodiment, each storage node 112 from
An example configuration for a selected storage enclosure 120 is shown in
In the context of an HDD, the storage media may take the form of one or more axially aligned magnetic recording discs which are rotated at high speed by a spindle motor.
Data transducers can be arranged to be controllably moved and hydrodynamically supported adjacent recording surfaces of the storage disc(s). While not limiting, in some embodiments the storage devices 122 are 3½ inch form factor HDDs with nominal dimensions of 5.75 in ×4 in ×1 in. or 2½ inch form factor HDDs with nominal dimensions of 100 mm×70 mm×7 or 9.5 mm.
In the context of an SSD, the storage media may take the form of one or more flash memory arrays made up of non-volatile flash memory cells. Read/write/erase circuitry can be incorporated into the storage media module to effect data recording, read back and erasure operations. Other forms of solid state memory can be used in the storage media including magnetic random access memory (MRAM), resistive random access memory (RRAM), spin torque transfer random access memory (STRAM), phase change memory (PCM), in-place field programmable gate arrays (FPGAs), electrically erasable electrically programmable read only memories (EEPROMs), etc.
In the context of a hybrid (SSHD) device, the storage media may take multiple forms such as one or more rotatable recording discs and one or more modules of solid state non-volatile memory (e.g., flash memory, etc.). Other configurations for the storage devices 122 are readily contemplated, including other forms of processing devices besides devices primarily characterized as data storage devices, such as computational devices, circuit cards, etc. that at least include computer memory to accept data objects or other system data.
The storage enclosures 120 include various additional components such as power supplies 124, a control board 126 with programmable controller (CPU) 128, fans 130, etc. to enable the data storage devices 122 to store and retrieve user data objects.
An example software architecture of the system 100 is represented by
The proxy server 136 accesses a plurality of map structures, or rings, to control data flow to the respective data storage devices 112 (
The account ring 140 provides lists of containers, or groups of data objects owned by a particular user (“account”). The container ring 142 provides lists of data objects in each container, and the object ring 144 provides lists of data objects mapped to their particular storage locations.
Each ring 140, 142, 144 has an associated set of services 150, 152, 154 and storage 160, 162, 164. The services and storage enable the respective rings to maintain mapping using zones, devices, partitions and replicas. The services may be realized by software, hardware and/or firmware. In some cases, the services are software modules representing programming executed by an associated processor of the system.
As discussed previously, a zone is a physical set of storage isolated to some degree from other zones with regard to disruptive events. A given pair of zones can be physically proximate one another, provided that the zones are configured to have different power circuit inputs, uninterruptable power supplies, or other isolation mechanisms to enhance survivability of one zone if a disruptive event affects the other zone. Contrawise, a given pair of zones can be geographically separated so as to be located in different facilities, different cities, different states and/or different countries.
Devices refer to the physical devices in each zone. Partitions represent a complete set of data (e.g., data objects, account databases and container databases) and serve as an intermediate “bucket” that facilitates management locations of the data objects within the cluster. Data may be replicated at the partition level so that each partition is stored three times, one in each zone. The rings further determine which devices are used to service a particular data access operation and which devices should be used in failure handoff scenarios.
In at least some cases, the object services block 154 an include an object server arranged as a relatively straightforward blob server configured to store, retrieve and delete objects stored on local storage devices. The objects are stored as binary files on an associated file system. Metadata may be stored as file extended attributes (xattrs). Each object is stored using a path derived from a hash of the object name and an operational timestamp Last written data always “wins” in a conflict and helps to ensure that the latest object version is returned responsive to a user or system request. Deleted objects are treated as a 0 byte file ending with the extension “.ts” for “tombstone.” This helps to ensure that deleted files are replicated correctly and older versions do not inadvertently reappear in a failure scenario.
The container services block 152 can include a container server which processes listings of objects in respective containers without regard to the physical locations of such objects. The listings may be as SQLite database files or some other form, and are replicated across a cluster similar to the manner in which objects are replicated. The container server may also track statistics with regard to the total number of objects and total storage usage for each container.
The account services block 150 may incorporate an account server that functions in a manner similar to the container server, except that the account server maintains listings of containers rather than objects. To access a particular data object, the account ring 140 is consulted to identify the associated container(s) for the account, the container ring 142 is consulted to identify the associated data object(s), and the object ring 144 is consulted to locate the various copies in physical storage. Commands are thereafter issued to the appropriate storage node 112 (
Additional services may be incorporated by or used in conjunction with the account, container and ring services 150, 152, 154 of
The system services can include replicators, updaters, auditors and a new storage management module. Generally, the replicators attempt to maintain the system in a consistent state by comparing local data with each remote copy to ensure all are at the latest version. Object replication can use a hash list to quickly compare subsections of each partition, and container and account replication can use a combination of hashes and shared high water marks.
The updaters attempt to correct out of sync issues due to failure conditions or periods of high loading when updates cannot be timely serviced. The auditors crawl the local system checking the integrity of objects, containers and accounts. If an error is detected with a particular entity, the entity is quarantined and other services are called to rectify the situation.
As explained below, the new storage management module carries out a variety of related functions responsive to the addition of new storage to the system. As used herein, the addition of new storage to the system can refer to (1) the replacement of an existing storage entity that has failed with a new, replacement storage entity; or (2) an expansion of the storage capacity of the storage cluster so that a larger number of storage entities are now in the system, thereby increasing the overall data storage capacity of the system.
With regard to the replacement of an existing storage entity that has failed, it will be appreciated that storage entities can be subjected to a failure condition from time to time and therefore require replacement. This is in contrast to temporary conditions, such as power outages, natural disasters, etc. where existing storage entities are simply “off-line” for a period of time but are subsequently brought back “on-line” after a service interruption and operate as before. Such storage entities can take a variety of hierarchical levels from individual drives 120, sets of drives, entire storage enclosures 120, sets of storage enclosures, entire storage assemblies 114 (storage cabinets), entire storage nodes, sets of storage nodes, and (in unlikely cases) an entire data center.
In the case of a temporary “off-line” situation, the new storage management module may or may not be involved. The replication of data objects within the system is intended to accommodate the temporary unavailability of two out of the three replicated sets, so in such cases normal processing may be carried out to continue data servicing until the off-line situation is corrected.
In the case of an actual failure event, the failed entity can be replaced with a new, replacement entity. For example, a data storage device 122 (e.g., HDD) may experience a failure condition and system administrative personnel may remove the failed HDD and replace it with a new, replacement HDD. Even if the new HDD has a larger storage capacity than the previous HDD (e.g., a 2 terabyte, TB (2×1012 bytes) drive is replaced by a 4 TB drive), the above described system services such as the replicators, updaters and auditors can operate to “reinstall” the data images from the failed entity onto the new, replacement entity without changes to the system mapping. In such case, the new storage management module may or may not be involved in such routine maintenance operations.
With regard to the provision of new storage capacity to the system, new storage entities can generally be supplied to expand the storage capacity of an existing storage node, or to add one or more new storage nodes to the system. In both cases, the new storage management module operates to detect the new storage entity or entities that have been added to the system, and to carry out data migration operations to redistribute data objects within the system to take advantage of the increase in overall available data capacity. Such data migration operations will generally tend to involve changes to the mapping (ring) structure(s) of the system.
The map data structure 180 is shown to include three primary elements: a list of devices 172, a partition assignment list 174 and a partition shift hash 176. The list of devices (devs) 172 lists all data storage devices 122 that are associated with, or that are otherwise accessible by, the associated ring, such as shown in Table 1.
Generally, ID provides an index of the devices list by device identification (ID) value. ZONE indicates the zone in which the data storage device is located. WEIGHT indicates a relative weight factor of the storage capacity of the device relative to other storage devices in the system. For example, a 2 TB (terabyte, 1012 bytes) drive may be given a weight factor of 2.0, a 4 TB drive may be given a weight factor of 4.0, and so on.
IP ADDRESS is the IP address of the storage controller associated with the device. TCP PORT identifies the TCP port the storage controller uses to serve requests for the device. DEVICE NAME is the name of the device within the host system, and is used to identify the disk mount point. METADATA is a general use field that can be used to store various types of arbitrary information as needed.
The partition assignment list 174 generally maps partitions to the individual devices. This data structure is a nested list: N lists of M+2 elements, where N is the number of replicas for each of M partitions. In some cases, the list 174 may be arranged to list the device ID for the first replica of each M partitions in the first list, the device ID for the second replica of each M partitions in the second list, and so on. The number of replicas N is established by the system administrators and may be set to three (e.g., N=3) or some other value. The number of partitions M is also established by the system administrators and may be a selected power of two (e.g., M=220, etc.).
The partition shift value 176 is a number of bits taken from a selected hash of the “account/container/object” path to provide a partition index for the path. The partition index may be calculated by translating a binary portion of the hash value into an integer number.
To service an access request upon the data objects stored in
The access command is forwarded to the associated storage node, and the local storage controller 108 (
A new, fourth cabinet 114 is shown in
Generally, the addition of the new storage entity (Cabinet D) to the storage node 112 results in the automated detection of the new storage, and in response thereto, the automated migration of some of the data from each of the existing entities (Cabinets A-C) to the new entity (Cabinet D), and the automated generation of a new map structure. The new map structure can be generated prior to, or after, the data migration, as discussed below.
Prior to the data migration, it can be seen that Storage 1 has a normalized utilization of about 70%, which means that only about 30% of the overall storage capacity of the entity is available to accommodate the storage of new data objects. Storage 2 has a normalized utilization of about 63% and Storage 3 has a normalized utilization of about 67%.
Once added to the system, Storage 4 receives a portion of the data stored in each of the first three entities Storage 1-3 so that all four of the storage entities have a nominally equal storage utilization, e.g., about 50%. It will be appreciated that the actual amounts of data migrated will be at object/partition boundaries, and various other considerations (including zoning) may come into play so that the final utilization distribution may not be equal across all of the devices. Nevertheless, the addition of the new storage capacity to the system operates to reduce the utilization of the existing entities and balance the amount of storage among the first and second sets of storage devices.
A second data distribution table 190 shows the data object sets D1-D8 after the data migration operation. It can be seen that some data object sets have been distributed to the available storage in the Storage 4 entity. A regular pattern of distribution can be applied, as represented in
In other cases, the data object sets D1-D8 can be sorted by size and the largest sets of data can be migrated first. In still other cases, a predetermined threshold can be assigned to the new storage (e.g., 40% utilization, etc.) and data migrations continue until the predetermined threshold is reached.
In some embodiments, the new storage detection block 202 operates in a plug-and-play manner to automatically detect the addition of new hardware and/or software applications. For example, the connecting of a new storage cabinet 114 as in
The new storage detection block 202 further operates to obtain existing mapping of the system and to generate new mapping that takes into account the additional capacity supplied by the new storage entities. In some cases, the data migration necessary to conform to the new mapping can take place prior to the deployment of the new mapping, so that the data migration is carried out “in the background” by the system. This approach is suitable for use, for example, when additional storage capacity is provided to a particular storage node and the additional capacity is going to be locally used by that node as opposed to significant transfers of data from other nodes to the new capacity.
When the data are migrated prior to mapping deployment, the data objects will already be stored in locations that substantially conform to the newly promulgated maps. In this way, other resources of the system such as set forth in
The new storage detection module 202 proceeds to generate a new map structure (“new mapping”), and supplies such to the data migration module 204. The map builder module 202 may further supply the existing mapping to the data migration module 204. As further depicted in
A migration sequencing block 210 schedules and directs the migration of data objects within the storage nodes 112 to conform the data storage state to the new mapping. This may include the issuance of various data migration commands to the respective storage nodes 112 in the system, as represented in
In response to the data migration commands, various data objects may be read, temporarily stored, and rewritten to different ones of the various entities in the storage node(s). It is contemplated that a substantial portion of the migrated data objects will be migrated from the “old” entities to the “new” entities, although for balancing purposes the data objects may also be moved between the old entities as well. The migration sequencing block 210 may receive command complete status indications from the node(s) signifying the status of the ongoing data migration effort.
It will be noted that while the data are being migrated, the data state will be intentionally placed in a condition where it deviates from the existing map structures of the respective nodes. In some cases, a transition management block 212 may communicate with other services of the system (e.g., the replicators, updaters, auditors, etc.) to suspend the operation of these and other system/ring services to not attempt to “undo” the migrations. An exception list, for example, may be generated and issued for certain data objects so that, should one or more of these services identify a mismatch between the existing mapping and the actual locations of the respective data objects, no corrective action will be taken. In other cases, the storage nodes affected by the data migrations may be temporarily marked as “off limits” for actions by these and other services until the completion of the migration. In still further embodiments, a special data migration command may be issued so that the storage controllers are “informed” that the migration commands are in anticipation of new mapping and therefore are intentionally configured to “violate” the existing mapping.
Once the data migration is complete, a migration complete status may be generated by the data migration module 204 and forwarded to the new storage detection block 202. Upon receipt of the migration complete status, the block 202 deploys the new mapping by forwarding new map structures to the various storage nodes in the system. At this point the data state of the system should nominally match that of the new map structures, and little if any additional overhead processing should be required by other system services to ensure conformance of the system to the newly deployed map structure(s).
Another issue that may arise from this processing is the handling of data access commands during the data migration process to accommodate the new storage. Depending on the size of the additional capacity added to the system, the processing depicted in
In such case, the migration sequencing block 210 may be configured to intelligently select the order in which data objects are migrated and/or “tombstoned” during the migration process. As the system maintains multiple replicas of every set of data objects (e.g., N=3), in some cases at least one set of data objects are maintained in an existing mapping structure so that data access commands can be issued to those replica sets of the data objects not affected by the migration. In some cases, these pristine replicas can be denoted as “source” replicas so that any access commands received during the data migration process are serviced from these replicas.
Additionally or alternatively, a temporary translation table can be generated so that, should the objects not be found using the existing mapping, the translation table can be consulted and a copy of the desired data objects can be returned from a cached copy and/or from the new location indicated by the new mapping.
While it is contemplated that the mapping will be deployed prior to data migration, such is not necessarily required. In an alternative embodiment, the new mapping may be generated and deployed by the new storage detection block 202, after which the data migration module 204 operates to conform the data object storage configuration to match the newly promulgated mapping. This latter approach is suitable, for example, when new storage nodes are added to the system. The device list for the new storage entities can be appended to the various ring structures by the new storage detection block 202, and the data migration module 204 can direct the rebalancing operation in an orderly fashion.
At step 222, various data objects supplied by users 138 of the system 100 are replicated in storage devices 122 housed in different zones in accordance with an existing mapping structure. The existing mapping structure may include the account, container and object rings 140, 142, 144 discussed above having a format such as set forth in
At some point during the operation of the system 100, new storage capacity is added to the system, as indicated by step 224. This will involve the connection of one or more new storage entities to the existing system to expand the overall storage capacity. The new storage capacity is detected at step 226 by the new storage detection block 202 of
At this point, the routine passes along one of two parallel paths. While either or both paths can be alternately used depending on system requirements, in some embodiments the first path is used when an existing storage node (or other storage pool) is expanded and the second path is used when new storage nodes (or other storage pools) are added to the system.
The first path includes identification at step 230 of the data objects that require migration to conform the system to the new map structure. This may be carried out by the data migration module 204 of
Although not shown in
Once the data migration is confirmed as being completed, step 234, the new map structures are deployed to the various storage nodes and other locations throughout the system at step 236, and the routine ends at step 238.
The second processing path involves deployment of the new storage mapping at step 240. This provides the new ring structures to each of the storage nodes and other operative elements in the system 100. The data migration module 204 proceeds to identify the data to be migrated in order to conform to the new mapping, step 242, the data are migrated at step 244, and as desired a data migration complete status can be generated at step 246.
Depending on the size of the additional storage capacity added to the system, significant amounts of data may be migrated internally in order to conform the system to the new mapping (whether prior to or after map deployment). Accordingly, further embodiments operate to identify the available bandwidth of the system 100 and the amount of data migration is throttled to ensure that sufficient resources are available to service the then-existing user I/O requirements.
The available bandwidth represents the data transfer capacity of the system 100 that is not currently being utilized to service data transfer operations with the users of the system. In some cases, the available bandwidth, BAVAIL, can be determined as follows:
B
AVAIL=(CTOTAL−CUSED)*(1−K) (1)
Where CTOTAL is the total I/O data transfer capacity of the system, CUSED is that portion of the total I/O data transfer capacity of the system that is currently being used, and K is a derating (margin) factor. The capacity can be measured in terms of bytes/second transferred between the proxy server 136 and each of the users 138 (see
The CUSED value can be obtained by the new storage management module 200 directly or indirectly measuring, or estimating, the instantaneous or average traffic volume per unit time at the proxy server 136. Other locations within the system can be measured in lieu of, or in addition to, the proxy server. Generally, however, it is contemplated that the loading at the proxy server 136 will be indicative of overall system loading in a reasonably balanced system.
The derating factor K can be used to provide margin for both changes in peak loading as well as errors in the determined measurements. A suitable value for K may be on the order of 0.02 to 0.05, although other values can be used as desired. It will be appreciated that other formulations and detection methodologies can be used to assess the available bandwidth in the system.
The available bandwidth BAVAIL may be selected for a particular sample time period TN. The sample time period can have any suitable resolution, such as ranging from a few seconds to a few minutes or more depending on system performance. Sample durations can be adaptively adjusted responsive to changes (or lack thereof) in system utilization levels.
The available bandwidth BAVAIL is provided to the data migration module 204, which selects an appropriate volume of data objects to be migrated during the associated sample time period TN. The volume of data migrated is selected to fit within the available bandwidth for the time period. In this way, the migration of the data will generally not interfere with ongoing data access operations with the users of the system. The process is repeated for each successive sample time period TN+1, TN+2, etc. until all of the pending data have been successfully migrated.
From a comparison of the relative heights of the respective cross-hatched areas 258, 262 in
The secondary path 272 is a second path between the respective storage devices of the respective storage cabinets 114. As with the primary data path, the secondary path 272 may also be one or more buses that interconnect the various storage enclosures within the cabinets 114 and therefore take the same general form as the primary path 270. Other forms can be used, such as fiber optic, wireless routing, coaxial cables, etc. for the secondary path 272. Suitable hardware (e.g., switches, etc.) and/or software (port control, etc.) may be added to the system to facilitate use of both paths 270, 272.
Generally, the system configuration of
It is contemplated that the secondary path can be permanently affixed to the cabinets 114 and used as a bypass path to migrate data objects between the respective cabinets. Alternatively, the secondary path 272 can be part of a new storage installation kit which is temporarily installed and used to migrate data between the respective cabinets 114, after which the pathway is removed once such migration is complete to accommodate the new Cabinet D.
The systems embodied herein are suitable for use in cloud computing environments as well as a variety of other environments. Data storage devices in the form of HDDs, SSDs and SSHDs have been illustrated but are not limiting, as any number of different types of media and operational environments can be adapted to utilize the embodiments disclosed herein
It is to be understood that even though numerous characteristics and advantages of various embodiments of the present disclosure have been set forth in the foregoing description, together with details of the structure and function of various embodiments thereof, this detailed description is illustrative only, and changes may be made in detail, especially in matters of structure and arrangements of parts within the principles of the present disclosure to the full extent indicated by the broad general meaning of the terms in which the appended claims are expressed.