Skip to content

Commit 7b9250d

Browse files
committed
Release 0.1.0
This is the initial release of the Predict library for crystal using the sgp4 model.
0 parents  commit 7b9250d

26 files changed

+4964
-0
lines changed

.gitignore

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
/doc/
2+
/libs/
3+
/lib/
4+
/.crystal/
5+
/.shards/
6+
7+
# Libraries don't need dependency lock
8+
# Dependencies will be locked in application that uses them
9+
/shard.lock
10+
11+
### C++ ###
12+
# Prerequisites
13+
*.d
14+
15+
# Compiled Object files
16+
*.slo
17+
*.lo
18+
*.o
19+
*.obj
20+
21+
# Precompiled Headers
22+
*.gch
23+
*.pch
24+
25+
# Compiled Dynamic libraries
26+
*.so
27+
*.dylib
28+
*.dll
29+
30+
# Fortran module files
31+
*.mod
32+
*.smod
33+
34+
# Compiled Static libraries
35+
*.lai
36+
*.la
37+
*.a
38+
*.lib
39+
40+
# Executables
41+
*.exe
42+
*.out
43+
*.app
44+
45+
/src/predict/ext/test
46+
/bench

.travis.yml

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
language: crystal

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
The MIT License (MIT)
2+
3+
Copyright (c) 2016 RX14
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in
13+
all copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21+
THE SOFTWARE.

README.md

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
# predict.cr
2+
3+
Predict is a satellite prediction library for crystal using the sgp4 model.
4+
The model used is the updated combined sgp/sdp4 model from the celestrak website.
5+
6+
Predict can track the latitude, longitude and altitude of satellites, and also calculate look angles (azimuth, elevation, range) from an observer on the earth's surface.
7+
8+
## Installation
9+
10+
Add this to your application's `shard.yml`:
11+
12+
```yaml
13+
dependencies:
14+
predict:
15+
github: RX14/predict.cr
16+
version: 0.1.0
17+
```
18+
19+
## Usage
20+
21+
```crystal
22+
require "predict"
23+
24+
# All values truncated to 4 decimal places for readability
25+
26+
# Parse TLE
27+
tle = Predict::TLE.parse_three_line <<-TLE
28+
ISS (ZARYA)
29+
1 25544U 98067A 16339.72355294 .00003337 00000-0 58323-4 0 9990
30+
2 25544 51.6456 287.6667 0006011 291.2532 140.8410 15.53794284 31519
31+
TLE
32+
33+
# Inspect TLE data
34+
tle.name # => "ISS (ZARYA)"
35+
tle.catalog_number # => 25544
36+
tle.mean_motion # => 15.5379
37+
38+
# Create satellite from TLE, this initialises the orbital model
39+
satellite = Predict::Satellite.new(tle)
40+
41+
# Create a prediction from the TLE
42+
time = Time.new(2016, 12, 5, 12, 0, 0)
43+
satellite_position, satellite_velocity = satellite.predict(time)
44+
45+
satellite_position # => Predict::TEME(@x=3815.7998, @y=1932.0874, @z=5261.6006)
46+
47+
# Predict ground track
48+
satellite_position.to_lat_long_alt(time) # => Predict::LatLongAlt(@latitude=0.8913, @longitude=2.3062, @altitude=415.4529)
49+
50+
# Predict look angles from an observer (200m altitude)
51+
observer = Predict::LatLongAlt.from_degrees(52.9, -2.24, 200.0)
52+
observer.look_at(satellite_position, time) # => Predict::LookAngle(@azimuth=0.50118495648349193, @elevation=-0.55812612380797122, @range=7501.0178628601843)
53+
```
54+
55+
## Development
56+
57+
SGP4 is bound as a static library in src/predict/ext. Use `make` with the optional make variables `release=true` or `debug=true` to build the static library. Make sure you run the specs and formatter before sending PRs.
58+
59+
Useful reading material on sgp4 and coordinate systems is http://celestrak.com/columns/.
60+
61+
## Contributing
62+
63+
1. Fork it ( https://github.com/RX14/predict.cr/fork )
64+
2. Create your feature branch (`git checkout -b feature/foo`)
65+
3. Commit your changes (`git commit`)
66+
4. Push to the branch (`git push origin feature/foo`)
67+
5. Create a new Pull Request
68+
69+
## Contributors
70+
71+
- [RX14](https://github.com/RX14) - creator, maintainer, confused guy

bench.cr

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
require "./src/predict"
2+
require "benchmark"
3+
4+
private def tle_name_zero
5+
<<-TLE
6+
0 AO-51 [+]
7+
1 28375U 04025K 09105.66391970 .00000003 00000-0 13761-4 0 3643
8+
2 28375 098.0551 118.9086 0084159 315.8041 043.6444 14.40638450251959
9+
TLE
10+
end
11+
12+
Benchmark.ips do |b|
13+
b.report("tle") { Predict::TLE.parse_three_line(tle_name_zero) }
14+
end

shard.yml

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
name: predict
2+
version: 0.1.0
3+
4+
authors:
5+
6+
7+
scripts:
8+
postinstall: make release=true -C src/predict/ext
9+
10+
license: MIT

spec/predict/coordinates_spec.cr

+95
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
require "../spec_helper"
2+
3+
private def gmst(time)
4+
Predict.greenwich_sidereal_time(time)
5+
end
6+
7+
describe Predict do
8+
describe ".greenwich_sidereal_time" do
9+
it "calculates the correct GMST" do
10+
gmst(Time.new(1995, 10, 1, 9, 0, 0)).should eq(2.5242182677688412)
11+
# From IAU sofa tests
12+
gmst(2400000.5 + 53736.0).should be_close(1.754174981860675096, 1e-12)
13+
end
14+
end
15+
16+
describe "TEME" do
17+
describe "#to_lat_long_alt" do
18+
it "converts TEME to lat/long/alt" do
19+
teme = Predict::TEME.new(-4400.594, 1932.870, 4760.712)
20+
expected = teme.to_lat_long_alt(Time.new(1995, 11, 18, 12, 46, 0))
21+
22+
expected.latitude.should be_close(44.91 * Predict::DEG2RAD, 0.5e-2)
23+
expected.longitude.should be_close(-92.31 * Predict::DEG2RAD, 0.5e-2)
24+
expected.altitude.should be_close(397.507, 0.5e-3)
25+
end
26+
27+
it "round trips" do
28+
time = Time.new(1995, 11, 18, 12, 46, 0)
29+
expected = Predict::TEME.new(-4400.594, 1932.870, 4760.712)
30+
actual = expected.to_lat_long_alt(time).to_teme(time)
31+
32+
actual.x.should be_close(expected.x, 1e-10)
33+
actual.y.should be_close(expected.y, 1e-10)
34+
actual.z.should be_close(expected.z, 1e-10)
35+
end
36+
37+
it "round trips (random)" do
38+
time = random_time
39+
expected = random_teme
40+
actual = expected.to_lat_long_alt(time).to_teme(time)
41+
42+
actual.x.should be_close(expected.x, 1e-10)
43+
actual.y.should be_close(expected.y, 1e-10)
44+
actual.z.should be_close(expected.z, 1e-10)
45+
end
46+
end
47+
end
48+
49+
describe "LatLongAlt" do
50+
describe "#to_teme" do
51+
it "converts latitude/longitude/altitude to TEME" do
52+
pos = Predict::LatLongAlt.from_degrees(40.0, -75.0, 0.0)
53+
teme = pos.to_teme(Time.new(1995, 10, 1, 9, 0, 0))
54+
55+
teme.x.should be_close(1703.295, 0.5e-3)
56+
teme.y.should be_close(4586.650, 0.5e-3)
57+
teme.z.should be_close(4077.984, 0.5e-3)
58+
end
59+
60+
it "round trips" do
61+
time = Time.new(1995, 10, 1, 9, 0, 0)
62+
expected = Predict::LatLongAlt.from_degrees(40.0, -75.0, 0.0)
63+
actual = expected.to_teme(time).to_lat_long_alt(time)
64+
65+
actual.latitude.should be_close(expected.latitude, 1e-10)
66+
actual.longitude.should be_close(expected.longitude, 1e-10)
67+
actual.altitude.should be_close(expected.altitude, 1e-10)
68+
end
69+
70+
it "round trips (random)" do
71+
time = random_time
72+
expected = random_lla
73+
actual = expected.to_teme(time).to_lat_long_alt(time)
74+
75+
actual.latitude.should be_close(expected.latitude, 1e-10)
76+
actual.longitude.should be_close(expected.longitude, 1e-10)
77+
actual.altitude.should be_close(expected.altitude, 1e-10)
78+
end
79+
end
80+
81+
describe "#look_at" do
82+
# it "finds look angles to a satellite" do
83+
# time = Time.new(1995, 11, 18, 12, 46, 0)
84+
# satellite = Predict::TEME.new(-4400.594, 1932.870, 4760.712)
85+
# observer = Predict::LatLongAlt.new(45.0, -93.0, 0.0)
86+
87+
# look_angle = observer.look_at(satellite, time)
88+
89+
# look_angle.azimuth.should be_close(100.36 * Predict::DEG2RAD, 0.5e-2)
90+
# look_angle.elevation.should be_close(81.52 * Predict::DEG2RAD, 0.5e-2)
91+
# look_angle.range.should be_close(81.52 * Predict::DEG2RAD, 0.5e-2)
92+
# end
93+
end
94+
end
95+
end

0 commit comments

Comments
 (0)